Java-数据结构-Map和Set(三)-习题 o(´^`)o

news2024/10/4 1:35:26

目录

❄️一、习题一(只出现一次的数字):

❄️二、习题二(随机链表的复制):

❄️三、习题三(宝石与石头):

 ❄️四、习题四(旧键盘):

 ❄️五、习题五(前k个高频单词):

 ❄️总结:


❄️一、习题一(只出现一次的数字):

         ☑ 题的传送门:

                      只出现一次的数字


这道题呢,有两种快速的解法,其一就是用 ^ 运算,其二就是使用 HashSet 来计算,我们一一来看,这些的解法:

其一:(^解法)

      我们呢在  相同的数字 ^  的时候呢就会为 0 ,所以我们的数字在 ^ 的时候呢,最后的出的数字就是我们的出现一次的数字,比如 1^1^2 = 2

代码:

class Solution {
    public int singleNumber(int[] nums) {
        int ret = nums[0];
        for(int i = 1;i < nums.length; i++) {
            ret = ret ^ nums[i];
        }
        return ret;
    }
}

 其一:(HashSet的解法)

         我们每次检查 HashSet 中呢是否有 nums[i] 这个数值,如果有呢就把这个数值从 HashSet中删除,如果没有这个数值,就把其入 HashSet 中,最后在 HashSet 中的数值就是我们要找的值

代码:

class Solution {
    public int singleNumber(int[] nums) {
        HashSet<Integer> set = new HashSet<>();
        for(int s : nums) {
            if(set.contains(s)) {
                set.remove(s);
            }else {
                set.add(s);
            }
        }
        for(int i = 0; i < nums.length;i++) {
            if(set.contains(nums[i])) {
                return nums[i];
            }
        }
        return -1;
    }
}

❄️二、习题二(随机链表的复制):

         ☑ 题的传送门:

                      随机链表的复制


    这道题呢不是简单得直接把 链表复制下来就结束的。这道题呢还是比较麻烦的,这个需要深拷贝的,这里呢,我们使用 HashMap 来做。

          因为呢 我们这道题的 random 可能指向的是自己也可能是空,也可能是 跳跃的指向。而且我们拷贝之后呢,我们新的节点是一个 新的地址 ,这样呢我们直接拷贝的话,我们的 next 和 random 这个就不是指向我们下一个节点了。

步骤: 

1、我们先把链表的节点的值拷贝到 HashMap 中

2、我们的 HashMap 里面存放的是 <Node,Node> 

3、根据我们 HashMap 中拷贝的节点值,再把对应的 next 和 random 拷贝进去。

 

OK,理解这个步骤之后呢,我们来看看代码如何编写的: 

class Solution {
    public Node copyRandomList(Node head) {
       HashMap<Node,Node> map  = new HashMap<>();
       Node cur = head;

       while (cur != null) {
        Node node = new Node(cur.val);
        map.put(cur,node);
        cur = cur.next;
       }

       cur = head;

       while(cur !=null) {
        map.get(cur).next = map.get(cur.next);
        map.get(cur).random = map.get(cur.random);
        cur = cur.next;
       }

       return  map.get(head);
    }
}

❄️三、习题三(宝石与石头):

         ☑ 题的传送门:

                        宝石与石头


       这个题呢还是比较简单的,我们直接使用 HashSet 就可以快速解决。

步骤:

1、我们先把 “宝石” 中的每一个字符放到 HashSet 中

2、遍历 “石头” 这个字符串,之后查看 HashSet 中是否存在 这个字符,如果存在 count++

是不是非常简单,所以呢我们来看看代码:

class Solution {
    public int numJewelsInStones(String jewels, String stones) {
        HashSet<Character> set = new HashSet<>();
        for(int i = 0;i < jewels.length();i++) {
            //先把宝石类型的字符串都放到HashSet中
            char ch = jewels.charAt(i);
            set.add(ch);
        }
        int count = 0;
        for(int j = 0;j < stones.length();j++) {
            //根据HashSet中的值和石头的字符一一比较,相同的count++
            char ch = stones.charAt(j);
            if(set.contains(ch)) {
                count++;
            }
        }
        return count;
    }
}

OK,这个题就是这个样子的非常的简单的,我们来看看下一道题。


 ❄️四、习题四(旧键盘):

         ☑ 题的传送门:

                      旧键盘


     这个题呢和我们上面的题呢是差不多的思路。

步骤:

1、先把 两个字符串 都转换成大写的

2、之后把 键盘坏的打的字符串 都放到 HashSet 中

3、我们再创建一个 HashSet ,遍历 好的键盘的打出来的字符 

4、如果开始的那个 HashSet 中没有 好的字符 并且 后创建的那个 HashSet 中没有这个字符的话,就将其打印出来,并且放到后面的 HashSet 中。

这个题呢,还是比较好理解的,所以呢我们直接来看代码如何编写的:

import java.util.Scanner;
import java.util.HashSet;
// 注意类名必须为 Main, 不要有任何 package xxx 信息
public class Main {
    public static void main(String[] args) {
        Scanner in = new Scanner(System.in);
        // 注意 hasNext 和 hasNextLine 的区别
        while (in.hasNextLine()) { // 注意 while 处理多个 case
            String str1 = in.nextLine();
            String str2 = in.nextLine();
            func(str1,str2);
        }
    }

    public static void func(String str1,String str2) {
        //转成大写
        str1 = str1.toUpperCase();
        str2 = str2.toUpperCase();

        //把坏的键盘的字符串放到 HashSet 中
        HashSet<Character> set1 = new HashSet<>();
        for(int i = 0; i < str2.length();i++) {
            char ch = str2.charAt(i);
            set1.add(ch);
        }

        HashSet<Character> set2 = new HashSet<>();        
        //遍历str1 看看set1 和 set2 中是否 存在
        //如果不存在就放到 set2 中 并且 打印
        for(int i = 0; i < str1.length();i++) {
            char ch = str1.charAt(i);
            if (!set1.contains(ch) && !set2.contains(ch)) {
                set2.add(ch);
                System.out.print(ch);
            }
        }
    }
}

OK,这个就是这个题的代码,上面的那个题麻烦一些。


 ❄️五、习题五(前k个高频单词):

         ☑ 题的传送门:

                       前k个高频单词


     这道题呢就比较难了,看到这个 “前k个” 是不是想起了我们以前介绍过的 Top-k 问题,我们的这道题呢就是存在 Top-k 问题的 并且和 HashMap 联合一起做这道题,才可以,我们来看看如何做到的。

步骤:

1、先定义一个 HashMap 里面的 key-value 是 String-Integer ,我们的 Integer 用来记录对应的String出现了几次。

2、建立 小根堆 这里不是简单的存储 Integer 而是存储 HashMap 的底层的 Map.Entry<String,Integer>,根据这里的Integer 来进行比较的,但是这里呢要注意的是,当不同的单词出现的次数相同的时候呢,我们根据 字典的顺序进行存储,所以这里的 创建小根堆 是要传一个新的比较器的

3、每次去 map 的数据,先把 堆的长度 和 k进行比较:

  如果:堆的长度  < k 的话,就直接入堆

  如果:堆的长度 >= k 的话,我们还需要判断:

             如果:堆顶数据的value值 < map的 value 值的话,说明map对应的 key 出现的次数多,                          就把 堆顶的数据进行出堆,并且把 map的这个数据 入堆。

             如果:堆顶数据的value值 == map的 value 值的话。我们就要判断对应的 key 的大小了

                        如果:堆顶数据的key 值 > map的 key 值的话,我们就把堆顶数据 出堆,并且把                                     map 的数据进行 入堆。

4、我们创建一个 ArrayList 的集合,把 堆中的 k 个 String 都放到 ArrayList 中,这个时候还是存在问题的,这时 ArrayList 中存放的是从小到大的排序,我们要把其 翻转过来。

5、对其 ArrayList 进行翻转。

这个呢就是这道题的大体思路了,我们来看看如何进行编写的代码:

class Solution {
   public List<String> topKFrequent(String[] words, int k) {
        HashMap<String,Integer> map = new HashMap<>();
        //1. 统计每个单词出现的次数
        for(String word : words) {
            if(map.get(word) == null) {
                map.put(word,1);
            }else {
                int val = map.get(word);
                map.put(word,val+1);
            }
        }

        //2. 建立小根堆
        PriorityQueue<Map.Entry<String,Integer>> minHeap = new PriorityQueue<>(new Comparator<Map.Entry<String, Integer>>() {
            @Override
            public int compare(Map.Entry<String, Integer> o1, Map.Entry<String, Integer> o2) {
                if(o1.getValue().compareTo(o2.getValue()) == 0) {
                    return o2.getKey().compareTo(o1.getKey());
                }
                return o1.getValue().compareTo(o2.getValue());
            }
        });
        //3.遍历map
        for(Map.Entry<String,Integer> entry : map.entrySet()) {
            if(minHeap.size() < k) {
                minHeap.offer(entry);
            }else {
                Map.Entry<String,Integer> top = minHeap.peek();
                if(top.getValue().compareTo(entry.getValue()) < 0) {
                    minHeap.poll();
                    minHeap.offer(entry);
                }else if(top.getValue().compareTo(entry.getValue()) == 0) {
                    if(top.getKey().compareTo(entry.getKey()) > 0) {
                        minHeap.poll();
                        minHeap.offer(entry);
                    }
                }
            }
        }
        ArrayList<String> list = new ArrayList<>();

        for (int i = 0; i < k; i++) {
            Map.Entry<String,Integer> tmp = minHeap.poll();
            list.add(tmp.getKey());
        }
        //e-2 b-5 c-6
        Collections.reverse(list);
        return list;
    }
}


 ❄️总结:

    OK,这次关于我们的 哈希表 相关的练习题呢,到这里就结束了,让我们下次再见,下次呢我们就要进入新的知识章节了,让我们尽情期待吧!!!拜拜~~~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2187167.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Nacos架构 原理】内核设计之Nacos一致性协议

文章目录 Nacos一致性协议为什么需要一致性协议Nacos选择了Raft&#xff08;强一致性&#xff09;&Distro&#xff08;最终一致性&#xff09;服务发现角度配置管理角度 Nacos自研Distro协议背景设计思想数据初始化数据校验写操作读操作 Nacos一致性协议 为什么需要一致性…

大模型笔记05--coze经典案例分析

大模型笔记05--coze经典案例分析 介绍经典案例分析抖音视频转小红书文案艺术照 & 卡通照片助手艺术照图像流卡通照片图像流多功能图像助手 注意事项说明 介绍 扣子是新一代 AI 应用开发平台&#xff0c;具备完善的生态系统&#xff0c;是国内最出色的AI平台之一。用好coze…

C/C++/EasyX ——入门图形编程(2)

【说明】这一篇的内容都是很基础的&#xff0c;所以内容会很多&#xff0c;具体现在也不知道要写多少&#xff0c;先写下去吧&#xff0c;新手小白们都不用担心&#xff0c;这个内容不会很难&#xff0c;因为我也是从一无所知过来的&#xff0c;很好入门的&#xff0c;&#xf…

游览器输入URL并Enter时都发生了什么 面试完美回答

文章目录 前言URL解析DNS解析**浏览器缓存****操作系统缓存**&#xff1a;**路由器缓存**&#xff1a;ISP&#xff08;Internet service provider&#xff09;缓存DNS递归解析IP地址的获取缓存结果 建立TCP连接发送HTTP请求服务器响应TCP链接断开渲染页面解析一 HTML解析过程解…

带你快速了解后端API服务的搭建

前言&#xff1a;写这篇文章的初衷是想分享一下我学习搭建后端API的过程&#xff0c;希望能帮助到和我一样想快速写API接口并部署到服务器上的同学&#x1f61c; 第一步&#xff1a;创建阿里云服务器 1、首先注册一个阿里云账号&#x1f917; 2、出于学习成本考虑&#xff0…

Linux系统,docker容器内查看pikachu源代码

在Linux系统中&#xff0c;要查看Docker容器内的Pikachu的源代码&#xff0c;需要先确保Pikachu的Docker镜像已经运行在系统上。以下是步骤和示例代码&#xff1a; 1、查找Pikachu容器的ID或名称&#xff1a; docker ps -a 2、使用docker exec命令进入运行中的Pikachu容器&am…

Spring MVC的运行流程详解

Spring MVC作为一个广泛使用的框架&#xff0c;提供了灵活且强大的MVC架构支持。尤其在业务系统中&#xff0c;Spring MVC能够有效地处理大量并发请求&#xff0c;提供良好的用户体验。本文将详细讲解Spring MVC的运行流程&#xff0c;以电商交易系统为案例&#xff0c;帮助读者…

不再烦恼!四款AI工具助你轻松打造完美PPT

嘿&#xff0c;各位办公室的小伙伴们&#xff0c;今儿咱们来聊聊那些让咱们工作生活大变样的“智能小伙伴”。作为每天跟PPT打交道的办公室文员&#xff0c;我敢说&#xff0c;自从有了这些神器&#xff0c;我的工作效率简直坐上了火箭&#xff0c;嗖嗖地往上涨&#xff01; 1…

Vue-Lecture1-Notes

渐进式框架 Vue 被称为“渐进式框架”&#xff0c;是因为它允许开发者根据项目的需求逐步引入和使用其功能&#xff0c;而不需要一次性使用整个框架。简单来说&#xff0c;Vue 提供了从简单到复杂的功能层次&#xff0c;可以灵活选择使用。 按需使用&#xff1a;Vue 的核心功能…

CSP-J Day 3 模拟赛补题报告

姓名&#xff1a;王胤皓&#xff0c;校区&#xff1a;和谐校区&#xff0c;考试时间&#xff1a; 2024 2024 2024 年 10 10 10 月 3 3 3 日 9 : 00 : 00 9:00:00 9:00:00~ 12 : 30 : 00 12:30:00 12:30:00&#xff0c;学号&#xff1a; S 07738 S07738 S07738 请关注作者的…

docker运行arm64架构的镜像、不同平台镜像构建

背景 Docker 允许开发者将应用及其依赖打包成一个轻量级、可移植的容器&#xff0c;实现“一次构建&#xff0c;到处运行”的目标。然而&#xff0c;不同的操作系统和硬件架构对容器镜像有不同的要求。例如&#xff0c;Linux 和 Windows 系统有不同的文件系统和系统调用&#…

银河麒麟桌面操作系统修改默认Shell为Bash

银河麒麟桌面操作系统修改默认Shell为Bash &#x1f490;The Begin&#x1f490;点点关注&#xff0c;收藏不迷路&#x1f490; 在银河麒麟桌面操作系统&#xff08;ARM版&#xff09;中&#xff0c;若要将默认Shell从Dash改为Bash&#xff0c;可执行以下步骤&#xff1a; 打开…

谈及医疗和教育真实需求是什么

医疗就是健康&#xff0c;物质基础身体棒&#xff1b; 教育就是心智&#xff0c;心理稳定智慧高。 提示&#xff1a;能让人内心充盈并不断提升自我能力的教育才是高品质的。 全文终结 需求量萎缩对品质端的改变-CSDN博客 与参考资料中的内容以及“贪嗔痴”的关联性进行详细扩…

【C++】vector类的模拟实现

✨ Blog’s 主页: 白乐天_ξ( ✿&#xff1e;◡❛) &#x1f308; 个人Motto&#xff1a;他强任他强&#xff0c;清风拂山冈&#xff01; &#x1f525; 所属专栏&#xff1a;C深入学习笔记 &#x1f4ab; 欢迎来到我的学习笔记&#xff01; 本篇文章参考博客&#xff1a;【C】…

HTML5+CSS+JavaScript剪子石头布游戏

HTML5CSSJavaScript剪子石头布游戏 用HTML5CSSJavaScript剪子石头布游戏实现剪子石头布游戏&#xff0c;游戏有成绩计数&#xff0c;人、机输赢情况&#xff0c;及平局情况。 ✂代表剪刀&#xff0c;▉代表石头&#xff0c;▓ 代表布&#xff0c;给出人机双方的出拳情况 游戏…

对于无人机行业技术、人才、实验环境共享,这事你怎么看?

对于无人机行业技术、人才、实验环境共享这一话题&#xff0c;我认为这是一个具有前瞻性和积极意义的趋势&#xff0c;对于推动无人机行业的健康发展具有重要意义。以下是我对此事的详细看法&#xff1a; 一、技术共享促进创新与发展 1. 加快技术创新&#xff1a;技术共享能够…

web前端面试中拍摄的真实js面试题(真图)

web前端面试中拍摄的真实js面试题&#xff08;真图&#xff09; WechatIMG258.jpeg WechatIMG406.jpeg WechatIMG407.jpeg WechatIMG922.jpeg WechatIMG1063.jpeg © 著作权归作者所有,转载或内容合作请联系作者 喜欢的朋友记得点赞、收藏、关注哦&#xff01;&#xff01;…

散度旋度拉普拉斯算子速记

哈密顿运算符号想象成一个矢量&#xff08;对x偏导&#xff0c;对y偏导&#xff0c;对z偏导&#xff09; 1、旋度就是这个矢量做点乘 2、散度就是矢量做叉乘 3、拉普拉斯算子就是哈密顿算符点乘哈密顿算符

【自然语言处理】(1) --语言转换方法

文章目录 语言转换方法一、统计语言模型1. 词向量转换2. 统计模型问题 二、神经语言模型1. 词向量化2. 维度灾难3. 解决维度灾难4. embedding词嵌入5. Word2Vec技术5.1 连续词袋模型&#xff08;CBOW&#xff09;5.2 跳字模型&#xff08;Skip-gram&#xff09; 总结 语言转换方…

[论文笔记]SGPT: GPT Sentence Embeddings for Semantic Search

引言 解码器Transformer的规模不断壮大&#xff0c;轻松达到千亿级参数。同时由于该规模&#xff0c;基于提示或微调在各种NLP任务上达到SOTA结果。但目前为止解码器Transformer还无法应用在语义搜索或语句嵌入上。 为了简单&#xff0c;下文中以翻译的口吻记录&#xff0c;比…