【代码随想录】刷题笔记——哈希表篇

目录

242. 有效的字母异位词

349. 两个数组的交集

202. 快乐数

1. 两数之和

454. 四数相加 II

383. 赎金信

15. 三数之和

18. 四数之和


242. 有效的字母异位词

思路

代码

class Solution {public boolean isAnagram(String s, String t) {if (s.length() != t.length()) return false;int[] cnt = new int[26];for (int i = 0;i < s.length();i++) {cnt[s.charAt(i) - 'a']++;}for (int i = 0;i < t.length();i++) {cnt[t.charAt(i) - 'a']--;}for (int i = 0;i < cnt.length;i++) {if (cnt[i] > 0) return false;}return true;}
}

API:

字符串求length:s.length() 

数组求length:cnt.length

取出 i 位置的元素:s.charAt(i)

代码(排序)

class Solution {public boolean isAnagram(String s, String t) {if (s.length() != t.length()) {return false;}char[] str1 = s.toCharArray();char[] str2 = t.toCharArray();Arrays.sort(str1);Arrays.sort(str2);return Arrays.equals(str1, str2);}
}

API:

字符串转字符数组:s.toCharArray()

比较两字符数组:Arrays.equals(str1, str2)

349. 两个数组的交集

思路

先把 nums1 改成 HashSet 形式,对 nums2 筛选出在 nums1 中的元素,并考虑对 nums2 中的元素去重,放入一个新的 HashSet 中,最后转为答案要求的形式。

代码(数组表示哈希表,有->1 ,无->0)

class Solution {public int[] intersection(int[] nums1, int[] nums2) {int[] temp = new int[1000];int index = 0;//存在不存在的问题int[] map = new int[1001];for (int i = 0;i < nums1.length;i++) {map[nums1[i]] = 1;}for (int i = 0;i < nums2.length;i++) {if (map[nums2[i]] == 1) {temp[index] = nums2[i];map[nums2[i]] = 0;index++;}}int[] result = Arrays.copyOfRange(temp,0,index);return result;}
}

代码(Set集合去重)

class Solution {public int[] intersection(int[] nums1, int[] nums2) {Set<Integer> set1 = new HashSet<>();for (int num : nums1) {set1.add(num);}Set<Integer> set2 = new HashSet<>();for (int num : nums2) {if (set1.contains(num)) {set2.add(num);}}int[] result = new int[set2.size()];int index = 0;for (int num : set2) {result[index++] = num;}return result;}
}

Collection集合的遍历方式

//1.使用增强for遍历集合
for(String s: c){System.out.println(s); 
}
//2. 使用lambda表达式,结合forEach方法
c.forEach(s->System.out.println(s)); 

202. 快乐数

思路

题目中说了会 无限循环,那么也就是说求和的过程中,sum会重复出现,这对解题很重要!

当我们遇到了要快速判断一个元素是否出现集合里的时候,就要考虑哈希法了

所以这道题目使用哈希法,来判断这个sum是否重复出现,如果重复了就是return false, 否则一直找到sum1为止。

判断sum是否重复出现就可以使用set集合

还有一个难点就是求和的过程,如果对取数值各个位上的单数操作不熟悉的话,做这道题也会比较艰难。

代码

class Solution {public boolean isHappy(int n) {Set<Integer> set = new HashSet<>();while(true) {int temp = getSum(n);if (temp == 1) return true;if (!set.contains(temp)) {set.add(temp);n = temp;}else {return false;}}}//计算每个位置上的数字的平方和private int getSum(int n) {int sum = 0;while (n > 0) {sum += (n%10) * (n%10);n /= 10;}return sum;}
}

新增的方法不能写到默认的public方法里

代码(官方)

class Solution {public boolean isHappy(int n) {Set<Integer> record = new HashSet<>();while (n != 1 && !record.contains(n)) {record.add(n);n = getNextNumber(n);}return n == 1;}private int getNextNumber(int n) {int res = 0;while (n > 0) {int temp = n % 10;res += temp * temp;n = n / 10;}return res;}
}

1. 两数之和

思路

代码

class Solution {public int[] twoSum(int[] nums, int target) {Map<Integer,Integer> map = new HashMap<>();int[] result = new int[2];for (int i = 0;i < nums.length;i++) {int key = target - nums[i];if (map.containsKey(key)) {result[0] = i;result[1] = map.get(key);break;}map.put(nums[i],i);}return result;}
}

代码(官方)

class Solution {public int[] twoSum(int[] nums, int target) {Map<Integer, Integer> hashtable = new HashMap<Integer, Integer>();for (int i = 0; i < nums.length; ++i) {if (hashtable.containsKey(target - nums[i])) {return new int[]{hashtable.get(target - nums[i]), i};}hashtable.put(nums[i], i);}return new int[0];}
}

454. 四数相加 II

思路

代码

class Solution {public int fourSumCount(int[] nums1, int[] nums2, int[] nums3, int[] nums4) {int cnt = 0;Map<Integer,Integer> map = new HashMap<>();//1. 把nums1 nums2相加的所有可能值存到hashmapfor (int i = 0;i < nums1.length;i++) {for (int j = 0;j < nums2.length;j++) {int temp = nums1[i] + nums2[j];if (map.containsKey(temp)) {int a = map.get(temp);map.remove(temp);map.put(temp,a + 1);}else {map.put(temp,1);}}}//2. 遍历nums3 和 nums4,对相加的所有结果判断hashmap中是否存在for (int i = 0;i < nums3.length;i++) {for (int j = 0;j < nums4.length;j++) {int temp2 = 0 - nums3[i] - nums4[j];if (map.containsKey(temp2)) {cnt += map.get(temp2);}}}return cnt;}
}

代码(官方)

class Solution {public int fourSumCount(int[] nums1, int[] nums2, int[] nums3, int[] nums4) {int res = 0;Map<Integer, Integer> map = new HashMap<Integer, Integer>();//统计两个数组中的元素之和,同时统计出现的次数,放入mapfor (int i : nums1) {for (int j : nums2) {int sum = i + j;map.put(sum, map.getOrDefault(sum, 0) + 1);}}//统计剩余的两个元素的和,在map中找是否存在相加为0的情况,同时记录次数for (int i : nums3) {for (int j : nums4) {res += map.getOrDefault(0 - i - j, 0);}}return res;}
}

Map集合API补充

// 遍历键
for (String key : map.keySet()) {System.out.println(key);
}// 遍历值
for (Integer value : map.values()) {System.out.println(value);
}// 获取默认值(若键不存在)
Integer defaultValue = map.getOrDefault("apple", 0); // 返回 0// 合并操作(Java 8+)
map.merge("apple", 5, Integer::sum); // 若键不存在则添加,存在则合并值

Java 8+ 新增默认方法

方法签名说明
V getOrDefault(Object key, V defaultValue)安全获取值(键不存在时返回默认值)
V putIfAbsent(K key, V value)键不存在时才插入
boolean remove(Object key, Object value)仅当键值都匹配时才移除
boolean replace(K key, V oldValue, V newValue)键值都匹配时才替换为新值
V replace(K key, V value)键存在时替换值
void replaceAll(BiFunction<? super K, ? super V, ? extends V> function)批量替换所有值
void forEach(BiConsumer<? super K, ? super V> action)遍历所有键值对
V compute(K key, BiFunction<? super K, ? super V, ? extends V> remappingFunction)动态计算新值(可删除或更新)
V computeIfAbsent(K key, Function<? super K, ? extends V> mappingFunction)键不存在时计算新值并插入
V computeIfPresent(K key, BiFunction<? super K, ? super V, ? extends V> remappingFunction)键存在时计算新值
V merge(K key, V value, BiFunction<? super V, ? super V, ? extends V> remappingFunction)合并新旧值(常用于计数)

383. 赎金信

代码

class Solution {public boolean canConstruct(String ransomNote, String magazine) {Map<Character,Integer> map = new HashMap<>();for (char s: ransomNote.toCharArray()) {map.merge(s,1,Integer::sum);}for (char m: magazine.toCharArray()) {if (map.containsKey(m)) {map.merge(m,1,(oldValue,newValue) -> oldValue - newValue);}}for (Integer v: map.values()) {if (v > 0) {return false;}}return true;}
}

map.merge(s,1,Integer::sum);       //相当于旧值 + 1

map.merge(m,1,(oldValue,newValue) -> oldValue - newValue);  // //相当于旧值 - 1

代码(官方)

class Solution {public boolean canConstruct(String ransomNote, String magazine) {// shortcutif (ransomNote.length() > magazine.length()) {return false;}// 定义一个哈希映射数组int[] record = new int[26];// 遍历for(char c : magazine.toCharArray()){record[c - 'a'] += 1;}for(char c : ransomNote.toCharArray()){record[c - 'a'] -= 1;}// 如果数组中存在负数,说明ransomNote字符串中存在magazine中没有的字符for(int i : record){if(i < 0){return false;}}return true;}
}

15. 三数之和

思路

代码(官方)

class Solution {//定义三个指针,保证遍历数组中的每一个结果//画图,解答public List<List<Integer>> threeSum(int[] nums) {//定义一个结果集List<List<Integer>> res = new ArrayList<>();//数组的长度int len = nums.length;//当前数组的长度为空,或者长度小于3时,直接退出if(nums == null || len <3){return res;}//将数组进行排序Arrays.sort(nums);//遍历数组中的每一个元素for(int i = 0; i<len;i++){//如果遍历的起始元素大于0,就直接退出//原因,此时数组为有序的数组,最小的数都大于0了,三数之和肯定大于0if(nums[i]>0){break;}//去重,当起始的值等于前一个元素,那么得到的结果将会和前一次相同if(i > 0 && nums[i] == nums[i-1]) continue;int l = i +1;int r = len-1;//当 l 不等于 r时就继续遍历while(l<r){//将三数进行相加int sum = nums[i] + nums[l] + nums[r];//如果等于0,将结果对应的索引位置的值加入结果集中if(sum==0){// 将三数的结果集加入到结果集中res.add(Arrays.asList(nums[i], nums[l], nums[r]));//在将左指针和右指针移动的时候,先对左右指针的值,进行判断//如果重复,直接跳过。//去重,因为 i 不变,当此时 l取的数的值与前一个数相同,所以不用在计算,直接跳while(l < r && nums[l] == nums[l+1]) {l++;}//去重,因为 i不变,当此时 r 取的数的值与前一个相同,所以不用在计算while(l< r && nums[r] == nums[r-1]){r--;} //将 左指针右移,将右指针左移。l++;r--;//如果结果小于0,将左指针右移}else if(sum < 0){l++;//如果结果大于0,将右指针左移}else if(sum > 0){r--;}}}return res;}
}

Arrays.asList() 和 List.of() 

// 使用 List.of()(推荐)
List<Integer> immutableList = List.of(1, 2, 3); // 不可变// 使用 Arrays.asList()
List<Integer> fixedSizeList = Arrays.asList(1, 2, 3); // 大小固定但元素可变// 若需要可变列表,建议显式转换
List<Integer> mutableList = new ArrayList<>(Arrays.asList(1, 2, 3));
mutableList.add(4); // 允许添加元素
场景推荐方法
创建不可变列表List.of()
需要存储 nullArrays.asList()
需修改元素(不增减大小)Arrays.asList()
Java 8 及以下版本Arrays.asList()

18. 四数之和

代码(官方)

class Solution {public List<List<Integer>> fourSum(int[] nums, int target) {List<List<Integer>> quadruplets = new ArrayList<List<Integer>>();if (nums == null || nums.length < 4) {return quadruplets;}Arrays.sort(nums);int length = nums.length;for (int i = 0; i < length - 3; i++) {if (i > 0 && nums[i] == nums[i - 1]) {continue;}if ((long) nums[i] + nums[i + 1] + nums[i + 2] + nums[i + 3] > target) {break;}if ((long) nums[i] + nums[length - 3] + nums[length - 2] + nums[length - 1] < target) {continue;}for (int j = i + 1; j < length - 2; j++) {if (j > i + 1 && nums[j] == nums[j - 1]) {continue;}if ((long) nums[i] + nums[j] + nums[j + 1] + nums[j + 2] > target) {break;}if ((long) nums[i] + nums[j] + nums[length - 2] + nums[length - 1] < target) {continue;}int left = j + 1, right = length - 1;while (left < right) {long sum = (long) nums[i] + nums[j] + nums[left] + nums[right];if (sum == target) {quadruplets.add(Arrays.asList(nums[i], nums[j], nums[left], nums[right]));while (left < right && nums[left] == nums[left + 1]) {left++;}left++;while (left < right && nums[right] == nums[right - 1]) {right--;}right--;} else if (sum < target) {left++;} else {right--;}}}}return quadruplets;}
}

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

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

相关文章

Python爬虫实战:研究messytables库相关技术

1. 引言 在当今数字化时代,互联网上存在着大量有价值的数据。然而,这些数据通常以不规则的格式存在,尤其是表格数据,可能包含复杂的表头、合并单元格、不规则布局等问题。传统的数据处理工具往往难以应对这些挑战。 网络爬虫技术可以帮助我们从网页上自动提取数据,而 mes…

Vue3的组件通信方式

通信方式适用层级数据流向复杂度Props/Emits父子组件单向/双向★☆☆v-model父子组件双向★☆☆Provide/Inject跨层级组件自上而下★★☆事件总线任意组件任意方向★★★Pinia/Vuex全局状态任意方向★★☆Refs模板引用父子组件父→子★☆☆作用域插槽父子组件子→父★★☆Web W…

创客匠人:大健康创始人IP如何用“社会责任”构建品牌护城河

一、商业与责任的失衡困局部分大健康IP将利润置于首位&#xff0c;甚至牺牲用户利益&#xff0c;导致品牌形象脆弱。某保健品公司因夸大宣传被曝光后&#xff0c;尽管销量曾达千万&#xff0c;却因缺乏社会认同&#xff0c;一夜之间崩塌&#xff0c;证明没有社会责任支撑的商业…

AI:机器人未来的形态是什么?

机器人未来的形态将受到技术进步、应用场景需求和社会接受度的综合影响&#xff0c;以下是对未来机器人形态的预测&#xff0c;涵盖技术趋势、设计方向和应用场景&#xff1a; 1. 形态多样化与通用化 人形机器人&#xff08;Humanoid Robots&#xff09;&#xff1a; 趋势&…

创建 UIKit 项目教程

一、打开 XCode&#xff0c;选择 iOS 下的 App&#xff0c;然后点 Next二、Interface 选择 Storyboard&#xff0c;然后点 Next三、删掉 Main.storyboard四、删掉 SceneDelegate.swift五、AppDelegate.swift 只保留第一个函数六、在 AppDelegate.swift 文件里的 application 函…

防爬虫君子协定 Robots.txt 文件

1.什么是robots.txt ? robots.txt是一个位于网站根目录的文本文件,用于指导搜索引擎爬虫如何访问和抓取网站内容。它遵循特定的语法规则,是网站与爬虫通信的重要工具。当搜索引擎访问一个网站时,它首先会检查该网站的根域下是否有一个叫做robots.txt的纯文本文件。Robots.…

浅谈 Python 中的 yield——生成器对象与函数调用的区别

我们来看这么一个例子&#xff1a; def greeter():name yield "你是谁&#xff1f;"yield f"你好&#xff0c;{name}"g greeter() print(next(g)) # → "你是谁&#xff1f;" print(g.send("张三")) # → "你好&#xf…

云端docker小知识

1、docker的三个关键概念image、container、dockerfile2、docker的container3、dockerfile4、docker制作image5、linux&#xff08;ubuntu&#xff09;安装docker&#xff08;步骤1和4&#xff09;6、docker基本命令docker images 查看全部镜像docker rmi -f 1e5f3c5b981a 删除…

【Elasticsearch】昂贵算法与廉价算法

在 Elasticsearch 里&#xff0c;“昂贵”并不单指“CPU 时间”&#xff0c;而是综合了 **CPU、内存、磁盘 I/O、网络传输** 以及 **实现复杂度** 的代价。下面把常见“昂贵算法”拆开说&#xff1a;1. **高计算密度的文本算法** • **match_phrase slop**&#xff08;带跨距…

深度学习-多分类

​开头摘要​​&#xff1a; 本文将深入探讨如何使用PyTorch实现基于Softmax回归的MNIST手写数字识别系统。从多分类问题的核心概念出发&#xff0c;详细解析​​One-Hot编码​​技术如何将类别标签向量化&#xff0c;剖析​​交叉熵损失函数​​的数学原理及其在训练中的优化机…

JVM 类加载过程

一、加载&#xff08;Loading&#xff09;目标&#xff1a;把字节码文件&#xff08;.class&#xff09;“读入 JVM”&#xff0c;生成类的 “半成品”&#xff08;Class 对象&#xff09;。Bootstrap ClassLoader&#xff08;启动类加载器&#xff09;&#xff1a;负责加载 JV…

通俗范畴论13 鸡与蛋的故事番外篇

通俗范畴论13 鸡与蛋的故事番外篇 在上一篇中,我们得到了鸡与蛋的Set局部小范畴如下: 鸡与蛋 SetSetSet 局部小范畴 如上图所示,每个鸡来自于一个蛋,每个蛋来自于一只鸡,如此循环,以至于无穷… 是的,假设鸡与蛋两个对象代表的集合,都是无穷集合,这个系统就没有问题…

记录跟随recyclerview滑动的指示器

老早之前做的一个功能&#xff0c;横向recyclerview滑动时&#xff0c;底部做跟随滑动指示器。今天代码不用了&#xff0c;记录下代码。<LinearLayoutandroid:layout_width"match_parent"android:layout_height"wrap_content"android:layout_marginTop&…

快速过一遍Python基础语法

前言 本文章是深度学习的前导课&#xff0c;对有编程基础的小伙伴更加的友好&#xff08;C、C&#xff09;&#xff0c;如果完全没有学过任何一门编程语言也没有关系&#xff0c;本文章不会涉及到晦涩难懂的原理&#xff0c;只是简单的带大家过一遍Python的基础语法。 下面的操…

[爬虫实战] 多进程/多线程/协程-异步爬取豆瓣Top250

相关爬虫知识点&#xff1a;[爬虫知识] 深入理解多进程/多线程/协程的异步逻辑 相关爬虫专栏&#xff1a;JS逆向爬虫实战 爬虫知识点合集 爬虫实战案例 逆向知识点合集 前言&#xff1a; 在之前文章中&#xff0c;我们深入探讨了多进程、多线程和协程这三大异步技术的工作…

Git系列--1.初始Git

一、背景 目录 一、背景 二、认识 三、如何在Linux上安装Git 3.1检测git是否存在和版本 3.2安装和卸载git 3.2.1Centos 3.2.2Ubuntu 四、基本操作 4.1创建本地仓库 4.2必须的配置项 4.3宏观认识基本分区 我们会根据需求不断更改我们的文件内容&#xff0c;但有时我们会…

QWidget的属性

QWidget的属性 windowOpacityAPI说明windowOpacity()获取不透明数值&#xff0c;返回float&#xff0c;取值为0.0到1.0&#xff0c;其中0.0为全透明&#xff0c;1.0为完全不透明setWindowOpacity()设置控件的不透明数值注意点&#xff1a;窗口不透明度的变化并非精确的&#xf…

【PTA数据结构 | C语言版】后缀表达式求值

本专栏持续输出数据结构题目集&#xff0c;欢迎订阅。 文章目录题目代码题目 请编写程序&#xff0c;求给定的后缀表达式的值。 输入格式&#xff1a; 输入在一行中给出一个非空后缀表达式&#xff0c;其中操作数为 int 型整数&#xff0c;操作符包括加、减、乘、除、取模。各…

装配式建筑4.0:当房子像汽车一样被“智造”

传统建筑方式&#xff0c;如同手工打造艺术品一般&#xff0c;大部分工作依赖现场施工&#xff0c;工人在建筑工地进行混凝土浇筑、砖块堆砌、钢筋绑扎等繁杂工作。这种方式受天气、工人技术水平等因素影响极大&#xff0c;不仅施工周期漫长&#xff0c;质量也参差不齐。据统计…

Go语言生态成熟度分析:为何Go还无法像Java那样实现注解式框架?

近年来&#xff0c;Go语言因其性能高效、部署简单、并发模型优秀等特性&#xff0c;成为云原生与微服务架构中的热门语言。然而&#xff0c;在实际的企业级项目开发中&#xff0c;开发者普遍会发现一个现象&#xff1a;Go的开发效率&#xff0c;尤其在快速构建中大型业务系统时…