【算法】排序+双指针——leetcode三数之和、四数之和

这篇具有很好参考价值的文章主要介绍了【算法】排序+双指针——leetcode三数之和、四数之和。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

【算法】排序+双指针——leetcode三数之和、四数之和,算法,算法
三数之和

(1)排序+双指针

  算法思路: 和之前的两数之和类似,我们对暴力枚举进行了一些优化,利用了排序+双指针的思路:

  我们先排序,然后固定⼀个数 a ,接着我们就可以在这个数后面的区间内,使用之前两数之和使用的算法,快速找到两个数之和和固定的a等于target即可。

  但是要注意,为了避免其中有重复的解:

  我们需要在找到⼀个结果之后, left 和 right 指针要跳过重复的元素;同时在使用完一次双指针算法之后,固定的 a 也要跳过重复的元素。

  算法实现过程:

  给定一个包含n个整数的数组nums,函数返回所有不重复的三元组[a, b, c],使得a + b + c = 0。函数首先对数组进行排序,然后使用双指针的方法来遍历数组。外层循环通过变量i遍历数组,内层循环通过变量left和right来找到满足条件的三元组。

  如果三数之和小于0,则将left右移一位;如果大于0,则将right左移一位;如果等于0,则将三个数加入结果数组ret,并继续移动left和right直到找到不重复的三元组。最后,返回结果数组ret。

【算法】排序+双指针——leetcode三数之和、四数之和,算法,算法

【算法】排序+双指针——leetcode三数之和、四数之和,算法,算法

【算法】排序+双指针——leetcode三数之和、四数之和,算法,算法

class Solution {
public:
    vector<vector<int>> threeSum(vector<int>& nums) {
        sort(nums.begin(),nums.end());//排序
        vector<vector<int>> ret;//定义一个二维数组
        int n=nums.size();
        for(int i=0;i<n;)
        {
            int left=i+1;
            int right=n-1;
            if(nums[i]>0) break;//如果最小值大于0,那结果一定大于0
            while(left<right)
            {
                int sum=nums[i]+nums[left]+nums[right];
                if(sum<0) left++;//三数之和小于0,left++
                else if(sum>0) right--;//三数之和大于0,right--
                else 
                {
                    ret.push_back({nums[i],nums[left],nums[right]});
                    left++;//找到了一组解后,left和right都要改变
                    right--;//避免重复的解
                    while(left<right&&nums[left]==nums[left-1]) left++;
                    while(left<right&&nums[right]==nums[right+1]) right--;
                }//跳过相同的值,也是为了避免重复的解
            }
            i++;//i也要跳过相同的值,而且i不可以越界,所以要i<n
            while(i<n&&nums[i]==nums[i-1]) i++;
        }
        return ret;
    }
};

时间复杂度:O(n2)
【算法】排序+双指针——leetcode三数之和、四数之和,算法,算法
                            
【算法】排序+双指针——leetcode三数之和、四数之和,算法,算法
四数之和

(1)排序+双指针

  实现方式和三数之和类似,固定两个数即可。

class Solution {
public:
    vector<vector<int>> fourSum(vector<int>& nums, int target) {
        vector<vector<int>> ret;//定义一个二维数组
        sort(nums.begin(),nums.end());//排序
        int n=nums.size();
        for(int i=0;i<n;)//固定第一个数
        {
            for(int j=i+1;j<n;)//固定第二个数
            {
                int left=j+1;
                int right=n-1;//后面的测试用例比较大,用long long
                long long aim=(long long)target-nums[i]-nums[j];
                while(left<right)
                {
                    int sum=nums[left]+nums[right];
                    if(sum>aim) right--;//如果四数之和大于目标值,right--
                    else if(sum<aim) left++;//如果四数之和小于目标值,left++
                    else 
                    {
                        ret.push_back({nums[i],nums[j],nums[left],nums[right]});
                        left++;//left++,right--,避免有重复的情况
                        right--;//跳过相同的数
                        while(left<right&&nums[left]==nums[left-1]) left++;
                        while(left<right&&nums[right]==nums[right+1]) right--;
                    }
                }
                j++;//跳过和第二个数相同的数
                while(j<n&&nums[j]==nums[j-1]) j++;
            }
            i++;//跳过和第一个数相同的数
            while(i<n&&nums[i]==nums[i-1]) i++;
        }
        return ret;
    }
};

时间复杂度:O(N3)
【算法】排序+双指针——leetcode三数之和、四数之和,算法,算法
                            文章来源地址https://www.toymoban.com/news/detail-643386.html

到了这里,关于【算法】排序+双指针——leetcode三数之和、四数之和的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【算法挨揍日记】day04——15. 三数之和、18. 四数之和

      15. 三数之和 https://leetcode.cn/problems/3sum/ 给你一个整数数组  nums  ,判断是否存在三元组  [nums[i], nums[j], nums[k]]  满足  i != j 、 i != k  且  j != k  ,同时还满足  nums[i] + nums[j] + nums[k] == 0  。请你返回所有和为  0  且不重复的三元组。 注意: 答案中不可以包含重复的三元

    2024年02月09日
    浏览(42)
  • (双指针 ) 18. 四数之和 ——【Leetcode每日一题】

    难度:中等 给你一个由 n 个整数组成的数组 nums ,和一个目标值 target 。请你找出并返回满足下述全部条件且 不重复 的四元组 [nums[a], nums[b], nums[c], nums[d]] (若两个四元组元素一一对应,则认为两个四元组重复): 0 = a, b, c, d n a 、 b 、 c 和 d 互不相同 nums[a] + nums[b] + nums[c

    2024年02月07日
    浏览(45)
  • (双指针 ) 15. 三数之和 ——【Leetcode每日一题】

    难度:中等 给你一个整数数组 nums ,判断是否存在三元组 [nums[i], nums[j], nums[k]] 满足 i != j 、 i != k 且 j != k ,同时还满足 nums[i] + nums[j] + nums[k] == 0 。请 你返回所有和为 0 且不重复的三元组。 注意 :答案中不可以包含重复的三元组。 示例 1: 输入:nums = [-1,0,1,2,-1,-4] 输出:

    2024年02月09日
    浏览(38)
  • 【每日一题Day266】LC18四数之和 | 排序+双指针

    四数之和【 LC18 】 给你一个由 n 个整数组成的数组 nums ,和一个目标值 target 。请你找出并返回满足下述全部条件且 不重复 的四元组 [nums[a], nums[b], nums[c], nums[d]] (若两个四元组元素一一对应,则认为两个四元组重复): 0 = a, b, c, d n a 、 b 、 c 和 d 互不相同 nums[a] + nums[b]

    2024年02月16日
    浏览(31)
  • [哈希专题]四数相加|赎金信|三数之和|四数之和

    此题思路:前两个数求和sum,sum1=0-后两数之和。判断sum1是否在之前出现过,此时用到哈希表,即判断sum1是否在前两个数中出现过。从而求得个数。 此题思路:定义两个指针left和right,遍历i,看三个位置的值是否为0。如果三个点值大于0,right--;如果小于0,left++。等于零即

    2024年01月17日
    浏览(34)
  • day7 四数相加 赎金信 三数之和 四数之和

    - 四数相加      - 不用去重,所以简单遍历就是     - 四个数,如果四重循环,就是O n^4,反正求和相加有几组数而已,俩俩一组先加起来,前俩个用map记下来,key是加和,value是出现组次数,再遍历第三、四个数组,找加和为0的,int ret = 0,去加上记录次数就好了 - 赎金信

    2024年02月14日
    浏览(83)
  • day07 四数相加Ⅱ 赎金信 三数之和 四数之和

    题目链接:454 四数相加Ⅱ 题意 4个整数数组nums1, nums2, nums3, nums4的长度均为n,有多少个元组(i,j,k,l)使得 nums[i]+nums[j]+nums[k]+nums[l]==0    (本题不包含去重的逻辑,i,j,k,l  可以相等,一组元素与一组元素之间的各个元素也可以相等) 暴力解法 会报如下超时错

    2024年01月24日
    浏览(38)
  • 【算法专题突破】双指针 - 四数之和(8)

    目录 1. 题目解析 2. 算法原理 3. 代码编写 写在最后: 题目链接:18. 四数之和 - 力扣(Leetcode)  这道题跟三数之和也是一样的, 题目很好理解,就是四个数的和等于target的情况, 且这四个数不能重复。 首先还是暴力解法: 排序 + 暴力枚举 + set去重 我们当然是用优化的解法

    2024年02月09日
    浏览(28)
  • 【算法专题突破】双指针 - 三数之和(7)

    目录 1. 题目解析 2. 算法原理 3. 代码编写 写在最后: 题目链接:15. 三数之和 - 力扣(Leetcode)  题目就是要找出和为0的不重复的三元组, 注意三元组的每个元素是得不同的位置,那不重复又是什么意思呢? 我们可以看第一个示例, 他找出了三个三元组,但是他最后只返回

    2024年02月09日
    浏览(30)
  • 力扣精选算法100题——四数之和(双指针专题)

    上一篇讲到(俩数之和and三数之和)这一篇我要来解析四数之和,四数之和建立在三数之和的基础上,我们需要熟练掌握三数之和的算法原理,如果大家三数之和还没弄清楚,请点击三数之和and二数之和链接即可看到。  三数之和和四数之和的题意其实都一样。 找到出四个数

    2024年01月19日
    浏览(31)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包