目标和问题(两数&三数&四数)

LeetCode目标和问题总结

目标和,n个数和为m

https://www.jianshu.com/p/d32d3800e739

典型的dfs问题。

1.两数之和

想到的思路:

1)暴力枚举

2)排序后,双指针。分别在开头和结尾向中间遍历。

没有想到的思路:

3)hash 思路,遍历一次即可。示例代码使用的是unordered_map

遍历到x,如果target-x在hash表中出现过,返回target-x的坐标,两个坐标即为结果。如果没有出现过,则把x加入到hash表中。

相似

15.三数之和

已写博客:https://www.jianshu.com/p/7f14788ab788

题目

给你一个包含 n 个整数的数组 nums,判断 nums 中是否存在三个元素 a,b,c ,使得 a + b + c = 0 ?请你找出所有和为 0 且不重复的三元组。

注意:答案中不可以包含重复的三元组。

 

示例 1:

输入:nums = [-1,0,1,2,-1,-4]
输出:[[-1,-1,2],[-1,0,1]]
示例 2:

输入:nums = []
输出:[]
示例 3:

输入:nums = [0]
输出:[]
 

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/3sum
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

题解

和为0且不重复。难点是不重复.

去重的难点:

  1. 按照顺序枚举

「不重复」的本质是什么?我们保持三重循环的大框架不变,只需要保证:

第二重循环枚举到的元素不小于当前第一重循环枚举到的元素;

第三重循环枚举到的元素不小于当前第二重循环枚举到的元素。

2)对于每一重循环而言,相邻两次枚举的元素不能相同,否则也会造成重复。【在一次循环里,一个位置的元素不能重复出现两次】举个例子,如果排完序的数组为

[0, 1, 2, 2, 2, 3]
^ ^ ^
我们使用三重循环枚举到的第一个三元组为 (0, 1, 2)(0,1,2),如果第三重循环继续枚举下一个元素,那么仍然是三元组 (0, 1, 2)(0,1,2),产生了重复。因此我们需要将第三重循环「跳到」下一个不相同的元素,即数组中的最后一个元素 33,枚举三元组 (0, 1, 3)(0,1,3)。

作者:LeetCode-Solution
链接:https://leetcode-cn.com/problems/3sum/solution/san-shu-zhi-he-by-leetcode-solution/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

    //分为三层循环,因为三个数之和。
    //第一层,确定一个数
    //第二层,两个指针,左指针通过遍历实现,右指针每次倒着找确定的那个数。
    //因为去重,所以每个位置不需要遍历第二个相同的数字,一二层循环去重,右指针因为位置元素值确定,因此不用特别写去重

可以用 双指针去改进。

class Solution {
public:
    vector<vector<int>> threeSum(vector<int>& nums) {
        //分为三层
        //第一层,确定一个数
        //第二层,两个指针,左指针通过遍历实现,右指针每次倒着找确定的那个数。
        //因为去重,所以每个位置不需要遍历第二个相同的数字,一二层循环去重,右指针因为位置元素值确定,因此不用特别写去重
        sort(nums.begin(),nums.end());
        vector<vector<int>> res;
        for(int i=0;i<nums.size();i++){
            if(i>0 && nums[i]==nums[i-1])//第一个位置不是相同的数
            continue;
            int left=i+1;
            int right = nums.size()-1;
            int red = -nums[i];
            if(left==right){
                break;
            }
            if(nums[i]>0){
                break;
            }
            for(;left<right;left++){
                if(left>i+1&&nums[left]==nums[left-1])//第二个位置不是相同的数
                    continue;
                while(nums[left]+nums[right]>red&&right>left)
                        right-=1;
                if(left==right)
                        break;
                if(nums[left]+nums[right]==red){ //相等了
                    vector<int> now={nums[i],nums[right],nums[left]};
                    res.push_back(now);
                    right = nums.size()-1;
                    // right-=1;//判断不等于之前的
                    continue;
                }           
            }

        }
        return res;
    }
};

16.最接近的三数之和

已写博客:https://www.jianshu.com/p/b35092672bf6

可以直接暴力,

无需去重,其实思路比三数之和还好想一点。

18.四数之和

给定一个包含 n 个整数的数组 nums 和一个目标值 target,判断 nums 中是否存在四个元素 a,b,c 和 d ,使得 a + b + c + d 的值与 target 相等?找出所有满足条件且不重复的四元组。

注意:答案中不可以包含重复的四元组。

 

示例 1:

输入:nums = [1,0,-1,0,-2,2], target = 0
输出:[[-2,-1,1,2],[-2,0,0,2],[-1,0,0,1]]
示例 2:

输入:nums = [], target = 0
输出:[]

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/4sum
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。
class Solution {
public:
    vector<vector<int>> fourSum(vector<int>& nums, int target) {
        //先排序
        //循环两遍,确定 三四个数
        //i从0开始,下一层从i+1,然后设置j=(i+1)+1 双指针
        sort(nums.begin(),nums.end());
        vector<vector<int>> res;
        if(nums.size()<4)
            return res;
        for(int i=0;i<nums.size();i++){//第一重循环
            if(nums[i]>target && nums[i]>0)//开始想剪枝,但是负值+负值会更小
                break;
            if(i>0 && nums[i]==nums[i-1])//去重
                continue;
            
            for(int j=i+1;j<nums.size();j++){ //第二重循环
                if(nums[j]>target-nums[i] && nums[j]>0)
                    break;
                if(j>i+1 && nums[j]==nums[j-1])
                    continue;
                int red = target-nums[i]-nums[j];

                //left,right 类似快排一样遍历
                int index = -1;
                int left = j+1;
                int right = nums.size()-1;
                // cout<<left<<right<<endl;
                while(left<right){
                    if (nums[left] + nums[right] > red) {
                        right--;
                    } else if (nums[left] + nums[right]<red) {
                        left++;
                    } else {
                        vector<int> tmp={nums[i], nums[j], nums[left], nums[right]};
                        res.push_back(tmp);
                        // 找到答案时,双指针同时收缩
                        right--;
                        left++;

                        // 去重逻辑应该放在找到一个四元组之后
                        while (right > left && nums[right] == nums[right + 1]) right--;
                        while (right > left && nums[left] == nums[left - 1]) left++;


                    }
                
                }
            }
        }
        return res;
    }
};
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,163评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,301评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,089评论 0 352
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,093评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,110评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,079评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,005评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,840评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,278评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,497评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,667评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,394评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,980评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,628评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,649评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,548评论 2 352

推荐阅读更多精彩内容