C++刷题笔记(10)——leetcode15、18、454

题目1:15.三数之和

在这里插入图片描述

解法:排序+双指针

解题过程:
题目要求找出不重复且和为0的元组,
如果枚举的三元组(a,b,c)满足a<=b<=c,那么就会减少(b,a,c)、(c,b,a)这样的重复,可以通过将数组元素按从小到大排序实现;
在数组的每一重循环中,相邻两次枚举的元素不能相同,否则也会造成重复,如数组{0,1,-1,-1,-2},使用三重循环枚举到的第一个三元组为{0,1,-1},如果第三重循环继续枚举下一个元素,那么得到的三元组仍为{0,1,-1}。
通过上述两步可以解决重复的问题,接下来就是遍历数组,找到符合条件的a、b、c。
在这里插入图片描述
定义一个下标left 定义在i+1的位置上,定义下标right 在数组结尾的位置上,在数组中找到 abc 使得a + b +c =0,a = nums[i]、b = nums[left]、c =nums[right]
在遍历数组的过程中:
(1)如果排序后第一个元素大于0,那么后边就不会有三数之和等于0;
(2)跳过重复元素避免重复;
(3)如果三数之和大于0,说明右边界太大,right左移
(4)如果三数之和小于0,说明左边界太小,left右移
(5)如果三数之和等于0,判断判断左界和右界是否和下一位置重复,去除重复解。并同时将left、right移到下一位置,寻找新的解

class Solution {
public:
    vector<vector<int>> threeSum(vector<int>& nums) {
        sort(nums.begin(), nums.end());
        vector<vector<int>> result;
        for (int i = 0; i < nums.size(); i++) {        //枚举a
            if (nums[i] > 0) {                         //排序后如果第一个元素已经大于零,返回结果
                return result;
            }
            if (i > 0 && nums[i] == nums[i - 1]) {     //去重,保证和上一次枚举的数不相同
                continue;
            }
            int right = nums.size() - 1;               //定义右指针c
            int target = -nums[i];                     //通过-a = b + c来确定符合条件的b、c
            for (int left = i + 1; left < nums.size(); left++) {               //定义左指针b
                if (left > i + 1 && nums[left] == nums[left - 1]) {            //去重复
                    continue;
                }
                while (left < right && nums[left] + nums[right] > target) {    //为了满足a<b<c,就要保证left在right的左侧
                    right--;
                }
                if (left == right) {                //指针重合,就不会有满足 a+b+c=0且b<c 的 c
                    break;
                }
                if (nums[left] + nums[right] == target) {   //满足 b + c = -a
                    result.push_back({ nums[i], nums[left], nums[right] });
                }
                //一轮for循环结束,左指针右移
            }
        }
        return result;
    }
};

代码随想录也提供也一种写法,原理是一样的,逻辑比较清晰,但是上边的更巧妙:
时间复杂度都是 O ( n 2 ) O(n^2) O(n2)

class Solution {
public:
    vector<vector<int>> threeSum(vector<int>& nums) {
        vector<vector<int>> result;
        sort(nums.begin(), nums.end());
        // 找出a + b + c = 0
        // a = nums[i], b = nums[left], c = nums[right]
        for (int i = 0; i < nums.size(); i++) {
            // 排序之后如果第一个元素已经大于零,那么无论如何组合都不可能凑成三元组,直接返回结果就可以了
            if (nums[i] > 0) {
                return result;
            }
            // 错误去重方法,将会漏掉-1,-1,2 这种情况
            /*
            if (nums[i] == nums[i + 1]) {
                continue;
            }
            */
            // 正确去重方法
            if (i > 0 && nums[i] == nums[i - 1]) {
                continue;
            }
            int left = i + 1;
            int right = nums.size() - 1;
            while (right > left) {
                // 去重复逻辑如果放在这里,0,0,0 的情况,可能直接导致 right<=left 了,从而漏掉了 0,0,0 这种三元组
                /*
                while (right > left && nums[right] == nums[right - 1]) right--;
                while (right > left && nums[left] == nums[left + 1]) left++;
                */
                if (nums[i] + nums[left] + nums[right] > 0) {
                    right--;
                    // 当前元素不合适了,可以去重
                    while (left < right && nums[right] == nums[right + 1]) right--;
                } else if (nums[i] + nums[left] + nums[right] < 0) {
                    left++;
                    // 不合适,去重
                    while (left < right && nums[left] == nums[left - 1]) left++;
                } else {
                    result.push_back(vector<int>{nums[i], nums[left], nums[right]});
                    // 去重逻辑应该放在找到一个三元组之后
                    while (right > left && nums[right] == nums[right - 1]) right--;
                    while (right > left && nums[left] == nums[left + 1]) left++;

                    // 找到答案时,双指针同时收缩
                    right--;
                    left++;
                }
            }

        }
        return result;
    }
};

题目2:18.四数之和

在这里插入图片描述

解法:排序+双指针

与三数之和解法类似,注意对比写法上的区别

三数之和的双指针解法是一层for循环num[i]为确定值,然后循环内有left和right下标作为双指针,找到nums[i] + nums[left] + nums[right] == 0。

四数之和的双指针解法是两层for循环nums[k] + nums[i]为确定值,依然是循环内有left和right下标作为双指针,找出nums[k] + nums[i] + nums[left] + nums[right] == target的情况

五数之和、六数之和等等都采用这种解法。

class Solution {
public:
    vector<vector<int>> fourSum(vector<int>& nums, int target) {
        vector<vector<int>> result;
        sort(nums.begin(), nums.end());   
        for (int i = 0; i < nums.size() - 3; i++) {  //i从下标0开始   i为确定的第一个数,还有三个数要确定,因此要nums.size()-3
            if (i > 0 && nums[i] == nums[i - 1]) {   //去重
                continue;
            }
            if ((long)nums[i] + nums[i + 1] + nums[i + 2] + nums[i + 3] > target) {   //确定nums[i]后,连续四数之和大于target则后面不会有符合条件的数,直接退出循环  long防止整型溢出
                break;
            }
            if ((long)nums[i] + nums[nums.size() - 3] + nums[nums.size() - 2] + nums[nums.size() - 1] < target) {
                continue;              //确定nums[i]后,剩下三个数无论取什么值都小于target,直接进入下一轮循环,nums[i+1]
            }
            for (int k = i + 1; k < nums.size() - 2; k++) {   //k定义在i+1的位置
                if (k > i + 1 && nums[k] == nums[k - 1]) {    //去重
                    continue;
                }
                if ((long)nums[i] + nums[k] + nums[k + 1] + nums[k + 2] > target) {   //确定nums[i]和nums[k]后
                    break;                                                            //连续四数之和大于target则剩下两个数不论取什么值不会有符合条件的数,直接退出循环
                }
                if ((long)nums[i] + nums[k] + nums[nums.size() - 2] + nums[nums.size() - 1] < target) {  //确定nums[i]和nums[k]后
                    continue;                                                                            //剩下两个数无论去什么值都小于target,直接进入下一轮循环,nums[j+1]
                }
                int left = k + 1, right = nums.size() - 1;                                //定义左右指针
                while (left < right) {                                                    //与三数之和类似
                    int sum = nums[i] + nums[k] + nums[left] + nums[right];
                    if (sum == target) {
                        result.push_back({ nums[i], nums[k], nums[left], nums[right] });
                        while (left < right && nums[left] == nums[left + 1]) {            //去重,直到left的右边 不等于 left
                            left++;
                        }
                        left++;   //还需要再向右移动一步
                        while (left < right && nums[right] == nums[right - 1]) {          //去重,同理
                            right--;
                        }
                        right--;
                    }
                    else if (sum < target) {
                        left++;
                    }
                    else {
                        right--;
                    }
                }
            }
        }
        return result;
    }
};

同样贴上代码随想录的双指针写法:

class Solution {
public:
    vector<vector<int>> fourSum(vector<int>& nums, int target) {
        vector<vector<int>> result;
        sort(nums.begin(), nums.end());
        for (int k = 0; k < nums.size(); k++) {
            if (k > 0 && nums[k] == nums[k - 1]) {    //去重
                continue;
            }
            for (int i = k + 1; i < nums.size(); i++) {
                // 正确去重方法
                if (i > k + 1 && nums[i] == nums[i - 1]) {
                    continue;
                }
                int left = i + 1;
                int right = nums.size() - 1;
                while (right > left) {
                    // nums[k] + nums[i] + nums[left] + nums[right] > target 会溢出
                    if (nums[k] + nums[i] > target - (nums[left] + nums[right])) {
                        right--;
                        // 当前元素不合适了,可以去重
                        while (left < right && nums[right] == nums[right + 1]) right--;
                        // nums[k] + nums[i] + nums[left] + nums[right] < target 会溢出
                    }
                    else if (nums[k] + nums[i] < target - (nums[left] + nums[right])) {
                        left++;
                        // 不合适,去重
                        while (left < right && nums[left] == nums[left - 1]) left++;
                    }
                    else {
                        result.push_back(vector<int>{nums[k], nums[i], nums[left], nums[right]});
                        // 去重逻辑应该放在找到一个四元组之后
                        while (right > left && nums[right] == nums[right - 1]) right--;
                        while (right > left && nums[left] == nums[left + 1]) left++;

                        // 找到答案时,双指针同时收缩
                        right--;
                        left++;
                    }
                }

            }
        }
        return result;
    }
};

对于几数之和这种问题,并不适合使用哈希法,因为要先找到符合条件的元素放进数组中,再进行去重,这样做非常耗时

题目3:454.四数相加Ⅱ

在这里插入图片描述

解法:哈希表

本题是使用哈希法的经典题目
解题思路:

将四个数组分为两组,Hashmap存一组nums1、nums2,另一组nums3、nums4和Hashmap进行比对。

1.首先定义 一个unordered_map,key放a和b两数之和,value 放a和b两数之和出现的次数。
2.遍历数组nums1、nums2,统计两个数组元素之和,和出现的次数,放到map中。
3.定义变量count,用来统计 a+b+c+d = 0 出现的次数。
4.在遍历数组nums3、nums4,找到如果 0-(c+d) 在map中出现过的话,就用count把map中key对应的value也就是出现次数统计出来。
5.最后返回统计值 count

class Solution {
public:
    int fourSumCount(vector<int>& nums1, vector<int>& nums2, vector<int>& nums3, vector<int>& nums4) {
        unordered_map<int, int> map; //key:a+b的数值,value:a+b数值出现的次数
        // 遍历数组nums1和数组nums2,统计两个数组元素之和,和出现的次数,放到map中
        for (int a : nums1) {
            for (int b : nums2) {
                map[a + b]++;   //得到所有A[i]+B[j]的值并存入哈希映射中
            }
        }
        int count = 0; // 统计a+b+c+d = 0 出现的次数
        // 在遍历数组nums3和数组nums4,找到如果 0-(c+d) 在map中出现过的话,就把map中key对应的value也就是出现次数统计出来。
        for (int c : nums3) {
            for (int d : nums4) {
                if (map.find(0 - (c + d)) != map.end()) {   //(a + b) - (c + d) = 0
                    count += map[0 - (c + d)];
                }
            }
        }
        return count;
    }
};

哈希表总结

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
引用中的代码片段是一个递归函数,名为beautifulArray,它接受一个整数n作为参数,并返回一个vector<int>类型的结果。这个函数用于生成一个长度为n的漂亮数组,漂亮数组满足以下条件:如果将其划分为两个任意长度的子数组,那么两个子数组的元素之和不相等。 函数的实现思路是通过递归,将问题划分为两个子问题,然后合并子问题的解。首先判断n是否等于1,如果等于1,则直接返回一个包含1的数组。否则,将n减1后的一半传入递归函数beautifulArray,得到一个漂亮数组res1。然后将n的一半传入递归函数beautifulArray,得到另一个漂亮数组res2。最后,将res1中的每个元素乘以2并减去1,然后与res2中的每个元素乘以2合并到结果数组res中。 引用中的代码片段是一个函数,名为translateNum,它接受一个整数num作为参数,并返回一个整数。这个函数用于计算将整数num翻译成字符串的不同方法数。 函数的实现思路是将整数num转换为字符串str,然后创建一个长度为str.size()-1的动态数组dp,用于记录不同位置的翻译方法数。然后初始化dp和dp[1]为1,表示前两位数字的翻译方法数。接下来,从第三位数字开始遍历字符串str,如果当前数字与前两位数字可以构成一个在10到25之间的整数,则将dp[i]的值设置为dp[i-1]+dp[i-2],表示当前位置的翻译方法数为前一位和前两位数字的翻译方法数之和。最后,返回dp最后一个元素的值,即为翻译整数num的不同方法数。 引用中的代码片段是一个函数,名为numberOfArithmeticSlices,它接受一个vector<int>类型的数组nums作为参数,并返回一个整数。这个函数用于计算数组nums中等差子序列的个数。 函数的实现思路是通过动态规划,创建一个与nums长度相同的动态数组dp,用于记录以每个位置为结尾的等差子序列的个数。然后遍历数组nums,从第三个元素开始,判断当前元素与前两个元素是否构成等差数列,如果是,则将dp[i]的值设置为dp[i-1]加1,表示以当前位置为结尾的等差子序列个数为前一位的等差子序列个数加1。最后,返回dp数组中所有元素的和,即为等差子序列的个数。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值