454.四数相加II

卡哥建议:本题是使用map巧妙解决的问题,好好体会一下 哈希法如何提高程序执行效率,降低时间复杂度,当然使用哈希法会提高空间复杂度,但一般来说我们都是舍空间换时间, 工业开发也是这样。

题目链接/文章讲解/视频讲解:https://programmercarl.com/0454.%E5%9B%9B%E6%95%B0%E7%9B%B8%E5%8A%A0II.html

做题思路:

暴力解法用4个for循环,会超时。为啥用map做,是因为我们可以把遍历4个数组abcd,变成遍历两个数组(a+b),(c+d)。第一个数组先计算a+b的和,把和存到容器中,在遍历第二个数组c+d的时候,再到容器中判断有没有我们想要的元素(0-(a+b))。因为这个题最后是输出多少个符合要求,所以除了看有没有出现我们想要的元素外,还要统计出现的次数,元素和次数对应,在map中是key对应着value。需要注意的是次数不是加1,要加value里记录的数,视频里卡哥讲了这点。

代码的输入和第6天的题代码输入差不多,我这里不写了。

 1 int fourSumCount(vector<int>& A, vector<int>& B, vector<int>& C, vector<int>& D) {
2 unordered_map<int, int> umap; //key:a+b的数值,value:a+b数值出现的次数
3 // 遍历大A和大B数组,统计两个数组元素之和,和出现的次数,放到map中
4 for (int a : A) {
5 for (int b : B) {
6 umap[a + b]++; //映射到了次数,然后++
7 }
8 }
9 int count = 0; // 统计a+b+c+d = 0 出现的次数
10 // 在遍历大C和大D数组,找到如果 0-(c+d) 在map中出现过的话,就把map中key对应的value也就是出现次数统计出来。
11 for (int c : C) {
12 for (int d : D) {
13 if (umap.find(0 - (c + d)) != umap.end()) {
14 count += umap[0 - (c + d)]; //次数要加value
15 }
16 }
17 }
18 return count;
19 }

15. 三数之和

卡哥建议:本题虽然和 两数之和 很像,也能用哈希法,但用哈希法会很麻烦,双指针法才是正解,可以先看视频理解一下 双指针法的思路,文章中讲解的,没问题 哈希法很麻烦。

题目链接/文章讲解/视频讲解:https://programmercarl.com/0015.%E4%B8%89%E6%95%B0%E4%B9%8B%E5%92%8C.html

做题思路:

这个题用哈希法就和上个题的思路差不多,就是去重不好处理。这道题目使用双指针法 要比哈希法高效一些。先排完序后,用for循环中的 i 代表 a,双指针的 left 代表 b,right 代表 c,这里卡哥视频讲解更清晰!其中去重要判断 num[i] = num[i+1],num[i] = num[i-1],用的是num[i] = num[i-1],比如(-1,-1,2),当我们从第一个 -1 看,那(-1,-1,2) 要算一个结果,从第二个 -1 开始向前看有没有-1,如果有的话,那直接 continue,从下一个位置 i 开始;对 left,right 也得像num[i] 那样判断,不同的是,在一个for 循环确定 i 下,双指针靠移动,就是如果重复了,那双指针移动去检查下一个是否重复。

代码:

 1 vector<vector<int>> threeSum(vector<int>& nums) {
2 vector<vector<int>> result; //结果集
3 sort(nums.begin(), nums.end()); //先排序
4 // 找出a + b + c = 0
5 // a = nums[i], b = nums[left], c = nums[right]
6 for (int i = 0; i < nums.size(); i++) {
7 // 排序之后如果第一个元素已经大于零,那么无论如何组合都不可能凑成三元组,直接返回结果就可以了
8 if (nums[i] > 0) {
9 return result;
10 }
11 // 错误去重a方法,将会漏掉-1,-1,2 这种情况
12 /*
13 if (nums[i] == nums[i + 1]) {
14 continue;
15 }
16 */
17 // 正确去重a方法
18 if (i > 0 && nums[i] == nums[i - 1]) {
19 continue;
20 }
21 int left = i + 1; //双指针的初始化
22 int right = nums.size() - 1;
23 while (right > left) { //b 不能等于 c
24 // 去重复逻辑如果放在这里,0,0,0 的情况,可能直接导致 right<=left 了,从而漏掉了 0,0,0 这种三元组
25 /*
26 while (right > left && nums[right] == nums[right - 1]) right--;
27 while (right > left && nums[left] == nums[left + 1]) left++;
28 */
29 if (nums[i] + nums[left] + nums[right] > 0) right--; //,缩小范围,right向前移动
30 else if (nums[i] + nums[left] + nums[right] < 0) left++; //扩大范围,left向后移动
31 else { //等于0的情况
32 result.push_back(vector<int>{nums[i], nums[left], nums[right]}); //把3个数放进结果集,3个数也在容器里,所以result一开始时定义的时候容器里有容器
33 //去重逻辑应该放在找到一个三元组之后,对b 和 c去重
34 while (right > left && nums[right] == nums[right - 1]) right--;
35 while (right > left && nums[left] == nums[left + 1]) left++;
36
37 // 找到答案时,双指针同时收缩
38 right--; //放进后,双指针继续同时移动找下一个区间,重新进入下一个while循环
39 left++;
40 }
41 }
42
43 }
44 return result;
45 }

18. 四数之和

     卡哥建议: 要比较一下,本题和 454.四数相加II 的区别,为什么 454.四数相加II 会简单很多,这个想明白了,对本题理解就深刻了。 本题 思路整体和 三数之和一样的,都是双指针,但写的时候 有很多小细节,需要注意,建议先看视频。 

题目链接/文章讲解/视频讲解:https://programmercarl.com/0018.%E5%9B%9B%E6%95%B0%E4%B9%8B%E5%92%8C.html

做题思路:

     这题要去重,还是用双指针法。用两个for循环确定k , i,剩下的和上一题的思路差不多。这里不能直接剪枝 num[k] > target,因为target 可能是负数,只有 target 和 num[k] 是正数,>0 的时候,才能直接剪枝。在num[k]+num[i] = target 的基础上去重后,再进行双指针的移动。

代码:

 1 vector<vector<int>> fourSum(vector<int>& nums, int target) {
2 vector<vector<int>> result;
3 sort(nums.begin(), nums.end());
4 for (int k = 0; k < nums.size(); k++) {
5 // 剪枝处理
6 if (nums[k] > target && nums[k] >= 0) {
7 break; // 这里使用break,统一通过最后的return返回
8 }
9 // 对nums[k]去重
10 if (k > 0 && nums[k] == nums[k - 1]) {
11 continue;
12 }
13 for (int i = k + 1; i < nums.size(); i++) {
14 // 2级剪枝处理
15 if (nums[k] + nums[i] > target && nums[k] + nums[i] >= 0) {
16 break;
17 }
18
19 // 对nums[i]去重
20 if (i > k + 1 && nums[i] == nums[i - 1]) {
21 continue;
22 }
23 int left = i + 1;
24 int right = nums.size() - 1;
25 while (right > left) {
26 // nums[k] + nums[i] + nums[left] + nums[right] > target 会溢出
27 if ((long) nums[k] + nums[i] + nums[left] + nums[right] > target) {
28 right--;
29 // nums[k] + nums[i] + nums[left] + nums[right] < target 会溢出
30 } else if ((long) nums[k] + nums[i] + nums[left] + nums[right] < target) {
31 left++;
32 } else {
33 result.push_back(vector<int>{nums[k], nums[i], nums[left], nums[right]});
34 // 对nums[left]和nums[right]去重
35 while (right > left && nums[right] == nums[right - 1]) right--;
36 while (right > left && nums[left] == nums[left + 1]) left++;
37
38 // 找到答案时,双指针同时收缩
39 right--;
40 left++;
41 }
42 }
43
44 }
45 }
46 return result;
47 }

代码随想录算法训练营第七天| LeetCode 454.四数相加II 15. 三数之和 18. 四数之和的更多相关文章

  1. 代码随想录算法训练营day07 | leetcode 454.四数相加II 383. 赎金信 15. 三数之和 18. 四数之和

    LeetCode 454.四数相加II 分析1.0 这个最直接暴力法,不过过于暴力了,害怕.jpg 失误 读题理解失误:题目要求的是四元组的个数,读完题到我这里成了输出四元组,悲哉 分析2.0 记录有 ...

  2. 代码随想录算法训练营day01 | leetcode 704/27

    前言   考研结束半个月了,自己也简单休整了一波,估了一下分,应该能进复试,但还是感觉不够托底.不管怎样,要把代码能力和八股捡起来了,正好看到卡哥有这个算法训练营,遂果断参加,为机试和日后求职打下一个 ...

  3. 代码随想录算法训练营day02 | leetcode 977/209/59

    leetcode 977   分析1.0:   要求对平方后的int排序,而给定数组中元素可正可负,一开始有思维误区,觉得最小值一定在0左右徘徊,但数据可能并不包含0:遂继续思考,发现元素分布有三种情 ...

  4. 代码随想录算法训练营day22 | leetcode 235. 二叉搜索树的最近公共祖先 ● 701.二叉搜索树中的插入操作 ● 450.删除二叉搜索树中的节点

    LeetCode 235. 二叉搜索树的最近公共祖先 分析1.0  二叉搜索树根节点元素值大小介于子树之间,所以只要找到第一个介于他俩之间的节点就行 class Solution { public T ...

  5. 代码随想录算法训练营day17 | leetcode ● 110.平衡二叉树 ● 257. 二叉树的所有路径 ● 404.左叶子之和

    LeetCode 110.平衡二叉树 分析1.0 求左子树高度和右子树高度,若高度差>1,则返回false,所以我递归了两遍 class Solution { public boolean is ...

  6. 代码随想录算法训练营day12 | leetcode 239. 滑动窗口最大值 347.前 K 个高频元素

    基础知识 ArrayDeque deque = new ArrayDeque(); /* offerFirst(E e) 在数组前面添加元素,并返回是否添加成功 offerLast(E e) 在数组后 ...

  7. 代码随想录算法训练营day10 | leetcode 232.用栈实现队列 225. 用队列实现栈

    基础知识 使用ArrayDeque 实现栈和队列 stack push pop peek isEmpty() size() queue offer poll peek isEmpty() size() ...

  8. 代码随想录算法训练营day06 | leetcode 242、349 、202、1

    基础知识 哈希 常见的结构(不要忘记数组) 数组 set (集合) map(映射) 注意 哈希冲突 哈希函数 LeetCode 242 分析1.0 HashMap<Character, Inte ...

  9. 代码随想录算法训练营day03 | LeetCode 203/707/206

    基础知识 数据结构初始化 // 链表节点定义 public class ListNode { // 结点的值 int val; // 下一个结点 ListNode next; // 节点的构造函数(无 ...

  10. 代码随想录算法训练营day24 | leetcode 77. 组合

    基础知识 回溯法解决的问题都可以抽象为树形结构,集合的大小就构成了树的宽度,递归的深度构成的树的深度 void backtracking(参数) { if (终止条件) { 存放结果; return; ...

随机推荐

  1. express实现批量删除和分页

    后端代码批量删除 // 批量删除 router.get('/manyDel', function (req, res) { let { ids } = req.query if (ids&&a ...

  2. Typora 1.6.7永久激活

    介绍Typora介绍 具体看上面的我就不多介绍了 接下来我们开始教程 需要的文件 Typora安装包 破解补丁包 安装包下载 破解补丁下载 接下来我们全部下载后获得一个安装包一个补丁 安装包直接安装就 ...

  3. go中的sync.pool源码剖析

    sync.pool sync.pool作用 使用 适用场景 案例 源码解读 GET pin pinSlow getSlow Put poolChain popHead pushHead pack/un ...

  4. 深入浅出Java多线程(二):Java多线程类和接口

    引言 大家好,我是你们的老伙计秀才!今天带来的是[深入浅出Java多线程]系列的第二篇内容:Java多线程类和接口.大家觉得有用请点赞,喜欢请关注!秀才在此谢过大家了!!! 在现代计算机系统中,多线程 ...

  5. FaceFusion:探索无限创意,创造独一无二的面孔融合艺术!

    FaceFusion:探索无限创意,创造独一无二的面孔融合艺术! 它使用先进的图像处理技术,允许用户将不同的面部特征融合在一起,创造有趣和令人印象深刻的效果.这个项目的潜在应用包括娱乐.虚拟化妆和艺术 ...

  6. C/C++ 命名空间引用知识

    标准命名空间 命名空间的使用 #define _CRT_SECURE_NO_WARNINGS #include <iostream> using namespace std; // 命名空 ...

  7. Spring一套全通6—注解编程

    百知教育 --- Spring系列课程 --- 注解编程 第一章.注解基础概念 1. 什么是注解编程 指的是在类或者方法上加入特定的注解(@XXX),完成特定功能的开发. @Component pub ...

  8. 关于laravel路由无法访问(nginx)

    laravel路由无法访问 被这个问题折腾了 好几次了,记录一下希望可以帮到后面的朋友 在laravel路由中配置好了之后无法访问的问题有可能是因为在本地服务配置的问题(我使用的是nginx服务器) ...

  9. Java - CodeForces - 1230A

    题目: Dawid有了 4 包糖果.第 i 包里面有 Ai 个糖果. Dawid想把这四包糖果送给两个朋友,能否让两个朋友收到相同数量的糖果?注意,不能拆开任何一包糖,不能把糖果留给自己或扔掉,四包糖 ...

  10. 国产数据库TiDB初体验:简单易用,快速上手

    最近开始关注国产数据库的发展,为了能从技术人员的角度来实际体验国产中目前最流行的TiDB数据库,从今天起,在官方公布的课程开始正面了解TiDB的设计理念. 看了2小时的入门课程介绍,总体来说,还是有不 ...