代码随想录算法训练营第六天| 454.四数相加II ● 383. 赎金信 ● 15. 三数之和 ● 18. 四数之和 ● 总结

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 效率更高!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值