四数相加II

力扣题目链接(opens new window)

给定四个包含整数的数组列表 A , B , C , D ,计算有多少个元组 (i, j, k, l) ,使得 A[i] + B[j] + C[k] + D[l] = 0。

为了使问题简单化,所有的 A, B, C, D 具有相同的长度 N,且 0 ≤ N ≤ 500 。所有整数的范围在 -2^28 到 2^28 - 1 之间,最终结果不会超过 2^31 - 1 。

例如:

输入:

  • A = [ 1, 2]
  • B = [-2,-1]
  • C = [-1, 2]
  • D = [ 0, 2]

输出:

2

解释:

两个元组如下:

  1. (0, 0, 0, 1) -> A[0] + B[0] + C[0] + D[1] = 1 + (-2) + (-1) + 2 = 0
  2. (1, 1, 0, 0) -> A[1] + B[1] + C[0] + D[0] = 2 + (-1) + (-1) + 0 = 0

初见分析

题目给我们提供了nums1~4四个长度相同输入数组,我们需要在每个数组中选择一个下标,组成一个含4个元素的元组

这四个下标对应的数组中的元素相加需要正好为0

似乎没有能够使用哈希表的点

存下标吗?下标元组按题意来算有n的四次方种

也不太可能用暴力法,那样需要四个循环,复杂度为n的四次方

思路

先声明,本题就是哈希法的经典题型。如何想到用哈希表是本题的关键。先梳理思路吧

由题意,我们需要在四个数组中找出相加等于零的情况出现的次数

注意,不需要给出对应的元素,只要记录等于零这种情况出现的次数

一种自然的想法是:

  1. int count = 0;
  2. for(a:A){
  3. for(b:B){
  4. for(c:C){
  5. for(d:D){
  6. int sum = a + b + c + d;
  7. if(sum ==0){
  8. count ++;
  9. }
  10. }
  11. }
  12. }
  13. }

时间复杂度n * n * n * n,不太行

反正肯定是要遍历数组的,能不能分开遍历?

于是,可以将AB分为一组,CD分为另一组

而且,肯定不能同时遍历这两组,要不然还是n * n * n * n

所以得分开计算遍历到的值的和

题目的要求是

  1. nums1[i]+nums2[i]+nums3[i]+nums4[i] = 0

设A遍历到的值为a,以此类推有b、c、d

那么:

  1. a+b+c+d = 0

先遍历AB数组,计算a+b,存入一个集合

这里有一种情况需要注意:

例如所给的数组中元素全为0,那么不同的0相加也算是不同的一种情况

所以计算a+b时还要记录其出现过的次数

因为需要记录两个值,所以使用map比较合适

然后遍历CD数组,计算c+d

那么这里就有使用哈希表的地方了,参考两数之和的处理方式

我们可以去集合中查询一个 target ,target = 0-(c+d)

如果 target 在集合中存在,那么就找到了一个满足条件的情况

不难看出,这里的target就是a+b的值

代码

  1. class Solution {
  2. public:
  3. int fourSumCount(vector<int>& nums1, vector<int>& nums2, vector<int>& nums3, vector<int>& nums4) {
  4. unordered_map<int, int> map;//创建一个map作为哈希表
  5. //遍历前两个数组A、B
  6. for(int n1 : nums1){//也可以写auto
  7. for(int n2 : nums2){
  8. // int sum = n1 + n2;
  9. map[n1 + n2]++;//在map中的sum位置标记1
  10. }
  11. }
  12. int count = 0;
  13. //遍历剩下的数组CD
  14. for(int n3 : nums3){
  15. for(int n4 : nums4){
  16. int sum = n3 + n4;
  17. int target = 0 - sum;
  18. auto iter = map.find(target);
  19. if(iter != map.end()){//找到target
  20. count += map[target];
  21. }
  22. }
  23. }
  24. return count;
  25. }
  26. };
注意点
1、遍历vector

可以使用c++中迭代器的风格去遍历,即:

  1. for(int n1 : nums1){//也可以写auto,让编译器自己去判断类型,当然题目已经给了vector的类型就直接写int
  2. for(int n2 : nums2){
  3. // int sum = n1 + n2;
  4. map[n1 + n2]++;//在map中的sum位置标记1
  5. }
  6. }

也可以传统一点(实际上这样也是用迭代器)

  1. for (it = nums1.begin(); it != nums1.end();it++) {
  2. cout << *it << " ";
  3. }
2、map插入数据
  1. #include <map>
  2. void printMap(map<int,int>&m)
  3. {
  4. for (map<int, int>::iterator it = m.begin(); it != m.end(); it++)
  5. {
  6. cout << "key = " << it->first << " value = " << it->second << endl;
  7. }
  8. cout << endl;
  9. }
  10. void test01()
  11. {
  12. //插入
  13. map<int, int> m;
  14. //第一种插入方式
  15. m.insert(pair<int, int>(1, 10));
  16. //第二种插入方式
  17. m.insert(make_pair(2, 20));
  18. //第三种插入方式
  19. m.insert(map<int, int>::value_type(3, 30));
  20. //第四种插入方式
  21. m[4] = 40;
  22. printMap(m);
  23. //删除
  24. m.erase(m.begin());
  25. printMap(m);
  26. m.erase(3);
  27. printMap(m);
  28. //清空
  29. m.erase(m.begin(),m.end());
  30. m.clear();
  31. printMap(m);
  32. }
  33. int main() {
  34. test01();
  35. system("pause");
  36. return 0;
  37. }

就用第四中就行,把map当成python中的字典

【LeetCode哈希表#5】四数相加II(map)的更多相关文章

  1. 【哈希表】leetcode454——四数相加II

    编号454:四数相加II 给定四个包含整数的数组列表 A , B , C , D ,计算有多少个元组 (i, j, k, l) ,使得 A[i] + B[j] + C[k] + D[l] = 0. 为 ...

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

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

  3. Leetcode 454.四数相加II

    四数相加II 给定四个包含整数的数组列表 A , B , C , D ,计算有多少个元组 (i, j, k, l) ,使得 A[i] + B[j] + C[k] + D[l] = 0. 为了使问题简单 ...

  4. Java实现 LeetCode 454 四数相加 II

    454. 四数相加 II 给定四个包含整数的数组列表 A , B , C , D ,计算有多少个元组 (i, j, k, l) ,使得 A[i] + B[j] + C[k] + D[l] = 0. 为 ...

  5. 代码随想录第七天| 454.四数相加II、383. 赎金信 、15. 三数之和 、18. 四数之和

    第一题454.四数相加II 给你四个整数数组 nums1.nums2.nums3 和 nums4 ,数组长度都是 n ,请你计算有多少个元组 (i, j, k, l) 能满足: 0 <= i, ...

  6. 【算法训练营day7】LeetCode454. 四数相加II LeetCode383. 赎金信 LeetCode15. 三数之和 LeetCode18. 四数之和

    [算法训练营day7]LeetCode454. 四数相加II LeetCode383. 赎金信 LeetCode15. 三数之和 LeetCode18. 四数之和 LeetCode454. 四数相加I ...

  7. 四数相加II & 赎金信 & 三数之和 & 四数之和

    一.四数相加Ⅱ 454. 四数相加 II 1.方法概述 首先定义一个map,key放a和b两数之和,value 放a和b两数之和出现的次数.遍历大A和大B数组,统计两个数组元素之和,和出现的次数,放到 ...

  8. LeetCode 454.四数相加 II(C++)

    给定四个包含整数的数组列表 A , B , C , D ,计算有多少个元组 (i, j, k, l) ,使得 A[i] + B[j] + C[k] + D[l] = 0. 为了使问题简单化,所有的 A ...

  9. LeetCode454. 四数相加 II

    题目 给定四个包含整数的数组列表 A , B , C , D ,计算有多少个元组 (i, j, k, l) ,使得 A[i] + B[j] + C[k] + D[l] = 0. 分析 关键是如何想到用 ...

  10. [Swift]LeetCode454. 四数相加 II | 4Sum II

    Given four lists A, B, C, D of integer values, compute how many tuples (i, j, k, l)there are such th ...

随机推荐

  1. 信创CPU与牙膏厂和按摩店CPU的简单对比

    信创CPU与牙膏厂和按摩店CPU的简单对比 摘要 周天时学习验证了SPEC2006的工具. 晚上时写完了第一稿简单的使用 因为 SPEC 完整跑完非常漫长. 我想了一下短平快还是通过使用一个简单的 r ...

  2. [转帖]HotSpot 虚拟机对象探秘

    https://www.cnblogs.com/xiaojiesir/p/15593092.html 对象的创建 一个对象创建的时候,到底是在堆上分配,还是在栈上分配呢?这和两个方面有关:对象的类型和 ...

  3. 行云部署成长之路--慢SQL优化之旅 | 京东云技术团队

    ​ 当项目的SQL查询慢得像蜗牛爬行时,用户的耐心也在一点点被消耗,作为研发,我们可不想看到这样的事.这篇文章将结合行云部署项目的实践经验,带你走进SQL优化的奇妙世界,一起探索如何让那些龟速的查询飞 ...

  4. css hover频繁闪烁

    今天遇见一个问题. 在鼠标放上 图片上的时候. 删除图标一直不停的闪烁. 我当时觉得很奇怪,父子关系的结构 不应该闪烁呀. 看了下html和css,发现子元素(要hover)的元素是绝对定位了的 于是 ...

  5. Linux挂载新磁盘到根目录

    1.添加磁盘到需要挂载的机器上2.lsblk查看硬盘挂载情况,sdb,sdc为我新挂载的磁盘 3.fdisk -l查看挂载之前的分区情况, 4.为新硬盘创建分区 fdisk /dev/sdb,终端会提 ...

  6. 手撕Vue-构建Vue实例

    前言 要想使用Vue必须先创建Vue的实例, 创建Vue的实例通过new来创建, 所以说明Vue是一个类, 所以我们要想使用自己的Vue, 就必须定义一个名称叫做Vue的类. 只要创建好了Vue的实例 ...

  7. 多智能体强化学习算法【三】【QMIX、MADDPG、MAPPO】

    相关文章: 常见多智能体强化学习仿真环境介绍[一]{推荐收藏,真的牛} 多智能体强化学习算法[一][MAPPO.MADDPG.QMIX] 多智能体强化学习算法[二][MADDPG.QMIX.MAPPO ...

  8. PE结构:VA&RVA&FOA 转换复习

    复习一下,不然会忘 1.imagebase 映像基地址 ,默认是0x400000 2.va 虚拟地址,载入OD后的地址,已经映射到内存的地址. 计算实际装入地址 VA imagebase (映像基址) ...

  9. C/C++ 监控磁盘与目录操作

    遍历磁盘容量: #include <stdio.h> #include <Windows.h> void GetDrivesType(const char* lpRootPat ...

  10. a标签download属性跨域问题

    1.如果是加载了非同源的内容,该属性将失效,等于导航功能 2.在服务端设置Content-Disposition,使用HTTP响应头Content-disposition进行处理 3.先下载数据文件, ...