4Sum - LeetCode

题目链接

Merge Two Sorted Lists - LeetCode

注意点

  • 和3Sum那道题一样

解法

解法一:在3Sum的基础上再加一层循环即可。时间复杂度为O(n^3)

class Solution {
public:
    using num_t = vector<int>;
    vector<vector<int>> fourSum(vector<int>& nums, int target) {
        int n = nums.size(),newTarget,i = n-1,j,l,r;
        vector<num_t> ans;
        if(n < 4) return ans;
        sort(nums.begin(),nums.end());
        while(i >= 3)
        {
            newTarget = target - nums[i];
            j = i-1;
            while(j >= 2)
            {
                l = 0;
                r = j-1;
                while(l < r)
                {
                    int sum = nums[l]+nums[r]+nums[j];
                    if(sum < newTarget) l++;
                    else if(sum > newTarget) r--;
                    else
                    {
                        ans.push_back({nums[i],nums[j],nums[l],nums[r]});
                        l++;
                        r--;
                        while (l<r && nums[l-1]==nums[l]) l++;     
                        while (l<r && nums[r+1]==nums[r]) r--;
                    }
                }
                j--;
                while(j >= 2 && nums[j+1]==nums[j]) j--;
            }
            i--;
            while(i >= 3 && nums[i+1]==nums[i]) i--;
        }
        return ans;
    }
};

解法二:看了评论得到的思路。先二重循环,将所有可能出现的两数之和的值map保存。然后再二重循环,计算两数之和,比较加上map中存的值之后是否等于target。时间复杂度为O(n^2)

class Solution {
public:
    using num_t = vector<int>;
    vector<vector<int>> fourSum(vector<int>& nums, int target) {
        vector<num_t> ret;
        int n = nums.size(),i,j;
        if(n < 4) return ret;
        unordered_map<int,vector<pair<int,int>>> myMap;
        for(i = 0;i < n;i++)
        {
            for(j = i+1;j < n;j++)
            {
                myMap[target - nums[i] - nums[j]].push_back({i,j});
            }
        }
        for(i = 0;i < n;i++)
        {
            for(j = i+1;j < n;j++)
            {
                int temp = nums[i] + nums[j];
                if(myMap.count(temp))
                {
                    for(auto p:myMap[temp])
                    {
                        if(p.first != i && p.first != j && p.second != i && p.second != j)
                        {
                        ret.push_back({nums[p.first],nums[p.second],nums[i],nums[j]});
                        }
                    }
                }
            }
        }
        n = ret.size();
        for(i = 0;i < n;i++)
        {
            sort(ret[i].begin(),ret[i].end());
        }
        sort(ret.begin(),ret.end());
        ret.erase(unique(ret.begin(),ret.end()),ret.end());
        
        return ret;
    }
};

注:虽然这道题时间复杂度为O(n2)但是去重所花的时间过多,导致最终所花时间甚至高于O(n3)
再注:去重所花时间是固定的,取决于输入的数组长度。所以在求nSum的问题的时候,随着n的增大,解法二更有优势。

小结

  • unique函数
  • map中key对应的value是唯一的,如果想一个key对应多个value可以像解法二一样,构造一个map<int,vector<int>>
posted @ 2019-01-31 15:29  闽A2436  阅读(184)  评论(0编辑  收藏  举报