LeetCode 18.四数之和

LeetCode 18.四数之和 C++

思路🧐:

由题意得,四个数组里面的整数相加需要得到target这个目标值,且结果不能重复 ,那么我们可以用排序+双指针 的方式进行解答,将该数组变为升序数组 ,然后固定左边两个数,从第二个固定的数下一位开始创建一个left指针,数组结尾创建一个right指针。

创建好指针后,我们将target减去固定的两个数得到变量ret,这样只需要考虑两个指针相加是否等于ret。

此时两个指针相加有三种情况

  1. left+right>ret,数组升序,那么left往右所有数与right相加都大于ret,则需要right--
  2. left+right<ret,数组升序,那么right往左所有数与left相加都小于ret,则需要left++
  3. left+right==ret,表示找到了,记录四元组,left++和right--。此时需要考虑去重问题,当left++值与left一样,则继续++,当right--与right值一样,则继续--。

当left>right时,该次循环结束,找到了固定最左边两数的四元组,接下来第二个固定的数++,并判断是否一样,一样就继续++,然后继续双指针遍历,直到第二个固定的数走完数组,就该第一个固定的数开始++和去重,反复循环,直到第一个固定的数遍历完数组。

代码🔎:

c++ 复制代码
class Solution {
public:
    vector<vector<int>> fourSum(vector<int>& nums, int target) {
        sort(nums.begin(),nums.end());
        int n = nums.size();
        vector<vector<int>> fourV;
        for(int i = 0; i < n;) //固定第一个数
        {
            for(int j = i + 1; j < n;) //固定第二个数
            {
                int left = j + 1;
                int right = nums.size() - 1;
                long long ret = (long long)target - nums[i] - nums[j]; //防止int溢出,所以用longlong
                while(left < right)
                {
                    if(nums[left] + nums[right] > ret) //>ret,right--
                        right--;
                    else if(nums[left] + nums[right] < ret) //<ret,left++
                        left++;
                    else
                    {
                        fourV.push_back({nums[left],nums[right],nums[i],nums[j]});
                        left++;
                        right--;
                        while(left < right && nums[left - 1] == nums[left]) //去重
                            left++;
                        while(left < right && nums[right + 1] == nums[right])
                            right--;
                    }
                }
                //固定数去重,这里防止漏判,所以先++
                j++;
                while(j < n && nums[j] == nums[j - 1])
                    j++;
            }
            i++;
            while(i < n && nums[i] == nums[i - 1])
                i++;
        }
        return fourV;
    }
};

时间复杂度:O (N 2LogN) 空间复杂度:O(N)

相关推荐
泽020213 分钟前
C++之STL--list
开发语言·c++·list
BUG收容所所长23 分钟前
二分查找的「左右为难」:如何优雅地找到数组中元素的首尾位置
前端·javascript·算法
itsuifengerxing1 小时前
python 自定义无符号右移
算法
猎板PCB厚铜专家大族1 小时前
高频 PCB 技术发展趋势与应用解析
人工智能·算法·设计规范
dying_man2 小时前
LeetCode--24.两两交换链表中的结点
算法·leetcode
yours_Gabriel2 小时前
【力扣】2434.使用机器人打印字典序最小的字符串
算法·leetcode·贪心算法
Dovis(誓平步青云)2 小时前
探索C++标准模板库(STL):String接口的底层实现(下篇)
开发语言·c++·stl·string
草莓熊Lotso2 小时前
【数据结构初阶】--算法复杂度的深度解析
c语言·开发语言·数据结构·经验分享·笔记·其他·算法
KyollBM3 小时前
【CF】Day75——CF (Div. 2) B (数学 + 贪心) + CF 882 (Div. 2) C (01Trie | 区间最大异或和)
c语言·c++·算法
feiyangqingyun3 小时前
Qt/C++开发监控GB28181系统/取流协议/同时支持udp/tcp被动/tcp主动
c++·qt·udp·gb28181