454.四数相加II
视频讲解:学透哈希表,map使用有技巧!LeetCode:454.四数相加II_哔哩哔哩_bilibili
先使用dict存储nums1和nums2中的元素及其和a+b,然后再遍历剩下两个数组,如果 0-c-d 存在于nums3和nums4, 也就是a+b+c+d=0,那就存入a+b出现的次数,因为会有重复的情况。
记住dict累加的这种写法,或者写个ifelse判断也可以,dict1[a+b] = dict1.get(a+b, 0) + 1
383. 赎金信
因为题目所只有小写字母,那可以采用空间换取时间的哈希策略, 用一个长度为26的数组还记录magazine里字母出现的次数。然后再用ransomNote去验证这个数组是否包含了ransomNote所需要的所有字母。
“为了不暴露赎金信字迹,要从杂志上搜索各个需要的字母,组成单词来表达意思”说明杂志里面的字母不可重复使用,所以每次用完就要减1。
其实在本题的情况下,使用map的空间消耗要比数组大一些的,因为map要维护红黑树或者哈希表,而且还要做哈希函数,是费时的!数据量大的话就能体现出来差别了。 所以数组更加简单直接有效!如果是数组的话就是list[ord(index)-ord('a')],相对位置存储
15. 三数之和
视频讲解:梦破碎的地方!| LeetCode:15.三数之和_哔哩哔哩_bilibili
这道题有点难,思路我写的双指针。先排个序,开始遍历数组,如果i对应值都大于0那后面都不用看了肯定不满足,然后left right指针分别从i+1和length-1开始往中间移动。题目要求组合不能重复,所以要对三元素都去重,对于最左边的i来说,和前面一个遍历过的i-1匹配,如果相等则说明已经找过对应的组合了则跳过,不能和后面i+1去比较,这样的话只是在对组合里元素去重复而已没有用。而且left不能和right相等,相等的话就指向同一元素也不满足题意。然后判断sum是否为0,<0说明太小,left变大一点, >0太大,right变小一点。最后对left和right去重也是,如果left后一个值和当下一样那就跳过,right和前一个值相同跳过,也可以节省点时间。
最后记得遍历完以后返回结果,还有要记得在=0的时候也要更新left和right在移动指针时候,这种小细节很容易不注意然后就出错了。
python两种排序办法,new = sorted(list),或者直接list.sort()
18. 四数之和
视频讲解:难在去重和剪枝!| LeetCode:18. 四数之和_哔哩哔哩_bilibili
算是上面那道题的延伸吧,在此基础上多套了一层for循环作为多加入的一个数在最左边
由于要去重和剪枝,而这道题不是=0,所以判断条件得变为第一层为
if nums[k] > target and target > 0 and nums[k] > 0: break;不能直接返回答案,可能后面还是会有=target得出现
if k > 0 and nums[k] == nums[k - 1]:continue
然后第二层为
if nums[k] + nums[i] > target and target > 0 and nums[k] + nums[i] > 0:break;
if i > k + 1 and nums[i] == nums[i - 1]: continue
后面的话就和上一题基本一致了,开始移动操作双指针
总结
一般来说哈希表都是用来快速判断一个元素是否出现集合里。
对于哈希表,要知道哈希函数和哈希碰撞在哈希表中的作用.
哈希函数是把传入的key映射到符号表的索引上。
哈希碰撞处理有多个key映射到相同索引上时的情景,处理碰撞的普遍方式是拉链法和线性探测法。
接下来是常见的三种哈希结构:
- 数组
- set(集合)
- map(映射)
使用map的空间消耗要比数组大一些,因为map要维护红黑树或者符号表,而且还要做哈希函数的运算。所以数组更加简单直接有效!
题目没有限制数值的大小,就无法使用数组来做哈希表了。
主要因为如下两点:
- 数组的大小是有限的,受到系统栈空间(不是数据结构的栈)的限制。
- 如果数组空间够大,但哈希值比较少、特别分散、跨度非常大,使用数组就造成空间的极大浪费。
关于set,C++ 给提供了如下三种可用的数据结构:(详情请看关于哈希表,你该了解这些! (opens new window))
- std::set
- std::multiset
- std::unordered_set
std::set和std::multiset底层实现都是红黑树,std::unordered_set的底层实现是哈希, 使用unordered_set 读写效率是最高的
使用数组和set来做哈希法的局限。
- 数组的大小是受限制的,而且如果元素很少,而哈希值太大会造成内存空间的浪费。
- set是一个集合,里面放的元素只能是一个key,而两数之和这道题目,不仅要判断y是否存在而且还要记录y的下标位置,因为要返回x 和 y的下标。所以set 也不能用。
map是一种<key, value>
的结构,本题可以用key保存数值,用value在保存数值所在的下标。所以使用map最为合适。
C++提供如下三种map::(详情请看关于哈希表,你该了解这些! (opens new window))
- std::map
- std::multimap
- std::unordered_map
std::unordered_map 底层实现为哈希,std::map 和std::multimap 的底层实现是红黑树。
同理,std::map 和std::multimap 的key也是有序的,不需要key有序,选择std::unordered_map 效率更高!