算法分享——《双指针》

文章目录

  • ✅[《移动零》](https://leetcode.cn/problems/move-zeroes/)
    • 🌹题目描述:
    • 🚗代码实现:
    • 😴代码解析:
  • ✅[《复写零》](https://leetcode.cn/problems/duplicate-zeros/)
    • 🌹题目描述:
    • 🚗代码实现:
    • 😴代码解析:
      • 🧛‍♀️复写操作:
      • 🚲查找最后一位:
  • ✅[《快乐数》](https://leetcode.cn/problems/happy-number/description/)
    • 🌹题目描述:
    • 🚗代码实现:
    • 😴代码解析:
      • 🍗扩展(鸽巢原理):
  • ✅[《盛最多水的容器》](https://leetcode.cn/problems/container-with-most-water/description/)
    • 🌹题目描述:
    • 🚗代码实现:
    • 😴代码解析:
  • ✅[《有限三角形的个数》](https://leetcode.cn/problems/valid-triangle-number/description/)
    • 🌹题目描述:
    • 🚗代码实现:
    • 😴代码解析:
  • ✅[《两数之和 II - 输入有序数组》](https://leetcode.cn/problems/kLl5u1/description/)
    • 🌹题目描述:
    • 🚗代码实现:
    • 😴代码解析:
  • ✅[《三数之和》](https://leetcode.cn/problems/3sum/description/)
    • 🌹题目描述:
    • 🚗代码实现:
    • 😴代码解析:
      • 🚁双指针遍历:
      • ⚓去重操作:
  • ✅[《四数之和》](https://leetcode.cn/problems/4sum/description/)
    • 🌹题目描述:
    • 🚗代码实现:
    • 😴代码解析:

《移动零》

🌹题目描述:

给定一个数组 nums,编写一个函数将所有 0 移动到数组的末尾,同时保持非零元素的相对顺序。
请注意 ,必须在不复制数组的情况下原地对数组进行操作。

示例 1:

输入: nums = [0,1,0,3,12]
输出: [1,3,12,0,0]

示例 2:

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

🚗代码实现:

class Solution {
public:void moveZeroes(vector<int>& nums) {int dest = -1, cur = 0;while(cur < nums.size()){if(nums[cur] != 0) swap(nums[++dest], nums[cur++]); // 处理非0元素else cur++;}}
};

😴代码解析:

本次代码的思路与我们在学习排序所了解到的快速排序原理类似。
在快排中,最重要的思路是**区域划分。本次题目也是这样。
image-20240901143456817
对于该题目来说,最后的目标是使得全部0在数组后面,那我们想要实现该操作肯定是需要去遍历整个数组。
那么遍历也会有被
遍历处理过的数据未被遍历处理的数据**。这种情况我们的做法一般是对一个数组进行区域划分

对于解决区域划分问题,我们可以使用双指针算法。要么我们定义一个left或一个right,也可以是cur或一个dest。
是否记得我们在学习数据结构中,对于带环链表的题解中,我们有用到过快慢指针,其实那也是一种双指针的算法。

通过题目要求,我们是需要不改变非0元素的排列顺序,只是将全部0元素放在数组末尾。而通过区域划分,我们又可以对已处理过的数据进行区域划分
image-20240901144421094

那么为了遍历整个数组,首先我们需要定义一个cur指针来进行遍历判断数据,再然后通过dest指针来实现区域划分,用来标记区域划分位置。

cur的作用:从左向右扫描数据,遍历数组
dest的作用:在已处理的区间内,指向非0元素的最后一个位置(实现区域划分)image-20240901144844539

所以对应的区域划分就为:
[0, dest] ——> 非0元素
[dest + 1, cur - 1] ——> 0元素
[cur, size() - 1] ——> 未处理的数据

下面我们就来看看代码是如何实现的:

  • 首先关于dest和cur的定义,由于dest均为非0元素,并且也是处理过的数据,因此我们刚开始初始化的时候需要将dest设置为-1.
    image-20240901145310131

  • 接下来在cur指向0的时候,因为要满足已处理的数据满足前部分为非0元素,后部分为0元素,所以我们只需要将cur继续遍历,这样0元素就是已经处理过的数据,也满足了[dest + 1, cur - 1]为0元素
    image-20240901145622536

  • 如果遇到cur指向不为0时,只需要将dest往后走一格再与cur指向的值进行交换,随后cur再往后挪动一格。同样也满足区域划分,
    image-20240901145917003

  • 再然后又遇到0,所以cur再往后挪动一格,指向3时再++dest和cur交换,随后cur再往后一格
    image-20240901150056264
    —————————————————————————————————
    image-20240901150127158

  • 随后再按照上述步骤进行交换,就能得到正确结果:

    image-20240901150258869

《复写零》

🌹题目描述:

给你一个长度固定的整数数组 arr ,请你将该数组中出现的每个零都复写一遍,并将其余的元素向右平移。

注意:请不要在超过该数组长度的位置写入元素。请对输入的数组 就地 进行上述修改,不要从函数返回任何东西。

示例 1:

输入:arr = [1,0,2,3,0,4,5,0]
输出:[1,0,0,2,3,0,0,4]
解释:调用函数后,输入的数组将被修改为:[1,0,0,2,3,0,0,4]

示例 2:

输入:arr = [1,2,3]
输出:[1,2,3]
解释:调用函数后,输入的数组将被修改为:[1,2,3]

🚗代码实现:

class Solution
{
public:void duplicateZeros(vector<int>& arr) {int n = arr.size();int flag = 0;int dest = 0;while (dest < n){if (arr[dest] == 0){flag += 2;}else{++flag;}if (flag >= n) break;dest++;}int cur = n - 1;if (flag > n){arr[cur--] = 0;dest--;}// 复写操作while (cur >= 0){if(arr[dest] == 0){arr[cur] = arr[dest];cur--;}arr[cur--] = arr[dest--];}}
};

😴代码解析:

对于这道题目,如果说可以新开辟一块数组供我们使用的话,是很容易想的也很容易实现。可是现在难就难如何在原地实现呢?

我们一开始可以试试正向遍历,但是这个方法很快就走不通了,因为我们会覆盖掉本来存在的数据,导致后面就全部都是错的。

所以正向的思路走不通,我们可以”掉头“试试。我们从反向进行遍历:

🧛‍♀️复写操作:

  • 如果是要从反向开始遍历,我们就需要先找到答案中最后一个数是多少,就比如示例1中的答案的最后一个数字是4,因此我们可以将dest先放在4原数组的4上,然后将cur放在最后一个数字上,开始从后向前遍历
    image-20240901155033895
  • 先将判断dest指向的数据是否为0,此时dest指向的数据是4,然后将dest指向的数据覆盖掉cur指向的数据
    image-20240901155501203
  • 再将cur--,dest也进行--操作
    image-20240901155538194
  • 此时dest指向的数据为0,那么cur要把指向的数据覆盖为0,并且把前一个数据也覆盖为0,cur总共要将自己指向的值覆盖两次。然后cur--还有dest--
    image-20240901155735899
  • 随后继续进行,dest指向的不为0,所以将cur指向的数据进行覆盖为dest指向的数据
    image-20240901155849809
  • 再进行--操作
    image-20240901155916944
  • 同理,直接覆盖再进行--
    image-20240901155950648
  • 此时dest指向的数据又为0,因此覆盖掉cur指向的数据和cur前一个的数据
    image-20240901160058716
  • 继续同时进行--操作
    image-20240901160211716
  • 再覆盖
    image-20240901160223908

至此复写操作结束,结果也和答案一致。

  1. 先判断 dest 位置的值
  2. 决定 cur 向前移动一步还是两步
  3. 判断一下 cur 是否已经结束为止
  4. dest–

🚲查找最后一位:

上述的操作,代表我们已经掌握了本题的解法,但是我们是通过答案知道复写完之后的最后一位的数字结果是4,但我们又该如何不通过答案找到数字呢?

其实我们如果用最原始的方法也是可以的,就是先计算数组有多少个元素,再通过数0的个数再乘2就可以找出那个所谓的最后一个数。

例如在示例1中,我们先遍历数组,访问到非0元素就+1,遇到0元素就+2,然后统计起来。
如果统计的结果 == 数组元素的个数。
那我们就找到了dest的,也就是复写完数组中的最后一个数。
image-20240901162149890

这样我们就可以再定义一个cur指针指向最后一个元素,继续正常实现上述的复写操作就可以了。

[!WARNING]

但是我们还会遇到一种情况,需要特别注意拿来讨论:
image-20240901162512527

此时我们在实现复写操作的时候,必须要进行一次判断,如果统计结果 > 元素个数,就需要先将最后一个元素替换为0,然后dest和cur再同时--,以这时候的状态为基准继续执行上述的复写操作:
image-20240901163006648

《快乐数》

🌹题目描述:

编写一个算法来判断一个数 n 是不是快乐数。

「快乐数」 定义为:

  • 对于一个正整数,每一次将该数替换为它每个位置上的数字的平方和。
  • 然后重复这个过程直到这个数变为 1,也可能是 无限循环 但始终变不到 1。
  • 如果这个过程 结果为 1,那么这个数就是快乐数。

如果 n快乐数 就返回 true ;不是,则返回 false

示例 1:

输入:n = 19
输出:true
解释:
1^2 + 9^2 = 82
8^2 + 2^2 = 68
6^2 + 8^2 = 100
1^2 + 0^2 + 0^2 = 1

示例 2:

输入:n = 2
输出:false

🚗代码实现:

class Solution 
{
public:int func(int n){int sum = 0;while(n){sum += (n % 10) * (n % 10);n /= 10;}return sum;}bool isHappy(int n) {int slow = n;int fast = func(n);while(fast != slow){slow = func(slow);fast = func(func(fast));}return slow == 1;}};

😴代码解析:

我们刚拿到该题目的时候可以尝试画图,当我们在画图的过程中,我们就会发现这个图像非常像我们之前所讲解的一种题型:
image-20240902140445904

上述我所想解释的是,当作为一个快乐数,如果一直不断计算下去,它的答案会一直为1,所以会一直在1进行循环。可能目前还不能彻底回忆起我们之前讲解的提醒,那我们再来看看非快乐数是怎么样的:
image-20240902141437685

诶,是不是有点感觉了?是不是特别像我们以前在学习数据结构链表的一道OJ题,题目是“判断是否为环形链表”呢?对于判断成环问题,我们用的算法就是——快慢指针。如果你清楚可以看看我的博客:链表OJ

所以现在我们就清楚了,具体快慢指针的算法我就不在这里赘述了。

我们只要**判断它们相遇的时候,当前的值是多少就可以。**判断是不是快乐数了

但是不同于链表,我们这里要额外实现一个函数,用来计算各位的平方和并且得出结果。这如同是一个快慢指针的转换,同逻辑但不同形式。

🍗扩展(鸽巢原理):

我们在这里仅仅是介绍了两种情况,但有没有一种情况是一只无限循环下去,不形成环的情况呢?

在这里我可以先总结:不存在不成环的情况!

具体的证明我们可以用鸽巢原理来证明。首先我们需要先简答了解一下鸽巢原理

假设 有 n 个巢穴 和 n+1 个鸽子,如果要将这 n+1 个鸽子分配完并且这 n 个巢穴都分满,则一定会保证有一个 巢穴 中的鸽子数一定大于1

那么对于该题目来说,就算是一个无符号整形,它的最大取值是4294967295。差不多说是4.2 * 10^9.假设我现在要将无符号整形的范围扩大,扩大到9999999999,也就是9 * 10^9,那么这个 9 * 10^9进行上述的函数操作号,得到的结果就是 9^2 * 10 = 810,所以你夸张扩大后的值最大也就810。也就是说你的4294967295之内的全部数字,不管怎么样都是只能在810之内的,也就是[1, 810]。

此时这个[1, 810]这810个数字就是“巢穴”,若是现在我们进行了811此操作,而且又存在一个数字很好运,将810个数字全部填满,那么再进行下一次操作时,它一定也会在这个[1, 810]之中,所以会出现重复的数!因此这就是为什么一定**不存在不成环的情况!**

《盛最多水的容器》

🌹题目描述:

给定一个长度为 n 的整数数组 height 。有 n 条垂线,第 i 条线的两个端点是 (i, 0)(i, height[i])

找出其中的两条线,使得它们与 x 轴共同构成的容器可以容纳最多的水。

返回容器可以储存的最大水量。

**说明:**你不能倾斜容器。

示例 1:

img

输入:[1,8,6,2,5,4,8,3,7]
输出:49 
解释:图中垂直线代表输入数组 [1,8,6,2,5,4,8,3,7]。在此情况下,容器能够容纳水(表示为蓝色部分)的最大值为 49。

示例 2:

输入:height = [1,1]
输出:1

🚗代码实现:

class Solution 
{
public:int maxArea(vector<int>& height) {int V = 0;int left = 0, right = height.size() - 1;while(left < right){V = max(V, (right - left) * min(height[left], height[right]));if(height[left] < height[right]) left++;else right--;}return V;}
};

😴代码解析:

该题目就是我们从小到大经常听到的“木桶原理”的变形,题目也很好理解。我们需要不断的比较的数值,最终来实现比较体积的大小。

当然,我们可以使用暴力解法,直接不断枚举进行比较,但是时间复杂度就是O(N^2)了,所以肯定不推荐。那么目前又是对应着一个“数组”,因此我们可以采取双指针的思路。

我们可以在0位置处定义一个左指针left,在最后的位置定义右指针right即可。然后在判断的时候,让当前对应的高度较小的下标进行++ 或 – 操作,直到两指针相遇,这样就使得时间复杂度为O(N)。

image-20240902151612110

因为height[right]所对应的高度小,所以让right-- 去左边找合适的值,但是也会伴随着“底”长度的减少:
image-20240902151920049

所以我们只需要不断 ++ --就好,然后每次比较V的最大值,并保留最大值即可。

《有限三角形的个数》

🌹题目描述:

给定一个包含非负整数的数组 nums ,返回其中可以组成三角形三条边的三元组个数。

示例 1:

输入: nums = [2,2,3,4]
输出: 3
解释:有效的组合是: 
2,3,4 (使用第一个 2)
2,3,4 (使用第二个 2)
2,2,3

示例 2:

输入: nums = [4,2,3,4]
输出: 4

🚗代码实现:

class Solution 
{
public:int triangleNumber(vector<int>& nums) {sort(nums.begin(), nums.end());int sum = 0, flag = nums.size() - 1;while(flag > 1){int left = 0, right = flag - 1;while(left != right){if(nums[left] + nums[right] > nums[flag]){sum += (right - left); // 精华所在--right;}else ++left;}--flag;}return sum;}
};

😴代码解析:

我们在上小学的时候就知道,一个三角形的构成条件,一定是满足存在任意两条边 > 第三条边(即 a + b > c),这样才能组成一个三角形。

对于这道题当然也有暴力解法,就是一遍一遍进行排列组合进行比对,但是这样时间复杂度高,效率低下。因此我们需要换一种思路来思考。

仔细想一想,对于该道题我们又是在一段数组上进行操作,那么毫无疑问——“双指针”。但是一开始的数据是乱序的,如果我们一上来就进行比较就会毫无章法,毫无逻辑。因此我们需要做好前置工作,先排序!
因此我们可以使用 C++的sort函数来进行排序,在参数里面传入vector的迭代器,begin( ) end( )

这样我们排完序之后,代码就变得直观可看了,数组也具有==单调性==了,以下我们会对数组的单调性来展开:

  • 首先,排完序后,最右边的数肯定是最大的,而最左边的数就是最小的。
  • 不妨先判断最小的数字,加上中间的数字中的某个数字,才能做到大于最大的数?
  • 找到中间的某个数A,发现A能做到加上最左边的数能够大于最右边的数,我们就可以得出,从A下标到起始位置的下标对应的全部数,加上A下标对应的数,肯定大于整个数组最大的那个数。下面我们就来证明这个结论

下面我们拿数据 nums = [2, 2, 3, 4, 5, 6] 来进行讲解:
image-20240903101758390

如图所示,对于一个已经排完序的数组,最左边一定是最小的数而最右边一定是最大的数,这时我们找到了一个数A,我们发现A加上min是大于max的,所以由于数组是有序的,因此min和A之间的全部数 都大于 min,所以既然min加上A都可以大于max,所以中间的也全部都可以。

但是我们仔细看看,其实3 4 5这三个数也可以组成一个三角形,所以我们的A数要不断移动的,max同时也要移动,所以我们就可以利用单调性的双指针转化为:
image-20240903102755748

flag:作为标记位,代表着三角形中最长的边
left:作为数组中最小的值,代表着三角形中最短的边
right:最为重要的值,代表着三角形中合适的数与left相加大于flag,初始化时在flag左边一位方便遍历全部数据

刚刚我们讲了,这种情况下right 到 left之间的全部数都满足。
接下来我们需要将right往左,因为right右边的值肯定大于当前的right的值,所以往右走肯定可以满足构成三角形,所以我们right往左走。看看还有没有与之匹配的三角形。
image-20240903103229872

这个时候就nums[left] + nums[right] == nums[flag],不能组成三角形,因此最有可能的原因就是left小了,看看还有没有大一点的值加上nums[right]可以大于nums[flag]的,所以left向右移动

image-20240903103407674

这个时候就正好满足三角形,但是随后right移动后两指针相遇了,我们就结束循环了。

但是还没有完,我们还需要让flag向左移动,因为可能还有满足最长边为5的,并且其它两条边加起来大于5的边,因此我们还需要继续判断,同时还要重置 left 和 right 的值。
image-20240903103801404

至此再不断重复前面的操作即可!!!

  1. a + b > c ——> right - left, right–
  2. a + b <= c ——> left++

《两数之和 II - 输入有序数组》

🌹题目描述:

给定一个已按照 升序排列 的整数数组 numbers ,请你从数组中找出两个数满足相加之和等于目标数 target

函数应该以长度为 2 的整数数组的形式返回这两个数的下标值*。*numbers 的下标 从 0 开始计数 ,所以答案数组应当满足 0 <= answer[0] < answer[1] < numbers.length

假设数组中存在且只存在一对符合条件的数字,同时一个数字不能使用两次。

示例 1:

输入:numbers = [1,2,4,6,10], target = 8
输出:[1,3]
解释:2 与 6 之和等于目标数 8 。因此 index1 = 1, index2 = 3 。

示例 2:

输入:numbers = [2,3,4], target = 6
输出:[0,2]

示例 3:

输入:numbers = [-1,0], target = -1
输出:[0,1]

🚗代码实现:

class Solution 
{
public:vector<int> twoSum(vector<int>& numbers, int target) {int left = 0, right = numbers.size() - 1;while(left != right){if(numbers[left] + numbers[right] > target) --right;else if(numbers[left] + numbers[right] < target) ++left;else return {left, right};}return {-1, -1}; // 照顾编译器,随便写数据即可}
};

😴代码解析:

该题目和上面《有限三角形的个数》相似,两道题都是利用排序好的数组的单调性来进行判断,我们也很容易就能得出该题的解法——”双指针“

我们可以在最左边和最右边定义两个指针,用来向左向右遍历,判断条件也很简单:

  1. numbers[left] + numbers[right] > target 这个时候由于有序数组的单调性,仅需要先将right进行减减操作

  2. numbers[left] + numbers[right] < target 这个时候由于有序数组的单调性,仅需要先将left进行加加操作

  3. numbers[left] + numbers[right] == target 这个时候就找到了对应的下标,然后利用vector的隐式类型转换返回即可。

《三数之和》

🌹题目描述:

给你一个整数数组 nums ,判断是否存在三元组 [nums[i], nums[j], nums[k]] 满足 i != ji != kj != k ,同时还满足 nums[i] + nums[j] + nums[k] == 0 。请你返回所有和为 0 且不重复的三元组。

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

示例 1:

输入:nums = [-1,0,1,2,-1,-4]
输出:[[-1,-1,2],[-1,0,1]]
解释:
nums[0] + nums[1] + nums[2] = (-1) + 0 + 1 = 0 。
nums[1] + nums[2] + nums[4] = 0 + 1 + (-1) = 0 。
nums[0] + nums[3] + nums[4] = (-1) + 2 + (-1) = 0 。
不同的三元组是 [-1,0,1] 和 [-1,-1,2] 。
注意,输出的顺序和三元组的顺序并不重要。

示例 2:

输入:nums = [0,1,1]
输出:[]
解释:唯一可能的三元组和不为 0 。

示例 3:

输入:nums = [0,0,0]
输出:[[0,0,0]]
解释:唯一可能的三元组和为 0 。

🚗代码实现:

class Solution
{
public:vector<vector<int>> threeSum(vector<int>& nums) {vector<vector<int>> vv;sort(nums.begin(), nums.end());int flag = nums.size() - 1;// 如果nums[flag] < 0,那nums[left]和nums[right]也都小于0,这样相加就没意义了while(flag > 1 && nums[flag] >= 0) { int target = 0 - nums[flag], left = 0, right = flag - 1;while(left < right) // 一定是要 left < right 可不能 left != right,这样会造成left > right{if(nums[left] + nums[right] > target) --right;else if(nums[left] + nums[right] < target) ++left;     else{vv.push_back({nums[left], nums[right], nums[flag]});// 找到数据后还要继续遍历!left++, right--;while(left < flag && nums[left] == nums[left - 1]) ++left;while(right >= 0 && nums[right] == nums[right + 1]) --right;}}--flag;// 一定要避免越界while(flag > 1 && nums[flag] == nums[flag + 1]) --flag;}return vv;}
};

😴代码解析:

区别于前面的《两数之和》,本题目有特别多的细节问题需要我们来处理,因为本题目不再是要求遇到满足的就返回,而是将全部满足的结果存放至vector容器中,再返回,而且存储答案的容器中还不能有重复的答案

如果以暴力的角度来看,我们当然可以定义三个指针一个一个值进行比对,然后将匹配的三个值存放在vector中,再接着我们可以用set容器来进行去重操作。说实话利用set容器进行去重这还好说,但是你直接暴力的话,这个算法还是有些挫的。

所以我们完全可以像上述《两数之和》和《有限三角形的个数》那样先排序成满足单调性的数组,再利用双指针进行遍历,这样的算法不仅时间复杂度低,效率也高,而且我们也可以不用set容器就可以完成去重的操作!但是针对于去重操作,这一点尤为难受,如果你刚拿到这道题想的就是 双指针 + 不用set来操作,那你一上手就会出现很多细节问题,诸如越界啦,甚至说right指针跑到left指针左边去了等等,所以接下来我们来展开好好说一说!

🚁双指针遍历:

该算法的实现和《有限三角形的个数》算法一样,先将数组进行排序,再定义left 和 right指针,用来遍历,定义flag来标记最后一个值,然后判断nums[left] + nums[right] 是否 等于nums[flag]

但是本题目不一样,我们要找到满足nums[left] + nums[right] + nums[flag] == 0的全部值,这种我们也可以转化成nums[left] + nums[right] == -nums[flag]然后剩下的内容就和《有限三角形的个数》解法一样了。

⚓去重操作:

首先我们来分析问题,我们就拿题目给的例子然后进行排序再修改下数据,往里添加几个0和1:
image-20240905113620828

和前面讲过的题目一样,只是我们这里是要让nums[left] + nums[right] == -nums[flag]即可,为了方便我们在这里定义target == -nums[flag]
1、nums[left] + nums[right] < target就让left++
2、nums[left] + nums[right] > target就让right--
3、nums[left] + nums[right] == target就让各个值push_back到答案容器vector中
这些就是基本双指针判断了,我也不细讲了,我们主要是来讲解去重操作:

  • 由图可知,此时的各个值满足nums[left] + nums[right] < target(-4 + 1 < -2),所以我们需要将left++,然后再继续判断:
    image-20240905114234868

  • 接下来又发现nums[left] + nums[right] > target(-1 + 1 > -2),所以我们需要将right--
    image-20240905114416002

  • 后面我们就不断进行操作,直到遇到这种情况:
    image-20240905114716562

  • 这个时候就非常完美,因为此时的nums[left] + nums[right] == target(-1 + 0 == -1),所以我们直接push_back就好了,但是接下来该怎么办呢?如果直接break掉,对于这个例子是可以的,但是万一这之中也有满足的条件呢?因此这种情况是不能break掉的,我们还需要继续判断,所以我们直接++left--right,但是对于他们俩,在进行left++right--之后,会发现还是为-1和0:
    image-20240905115146197

    • 所以我们其实可以直接跳过这些,因为再进行一次判断也是浪费时间,所以直接用 while循环来处理就好,但是这里一定要注意喔,你可不要对 right 一直不断遍历判断然后就把right搞越界了,所以我们还需要先进行越界问题的处理:
      while(left < flag && nums[left] == nums[left - 1]) ++left
      while(right >=0 && nums[right] == nums[right + 1] --right)

    • 最后处理完之后 right 就会在 left 的左侧了,所以当前的这一趟循环就结束了,然后就会flag--进入下一趟:
      image-20240905115632622

    • 可是这个flag还是和之前一样啊,那你判断的话岂不是又要再进行一次判断了?那你百分之百又会遇到那几个数字,那你在存放答案的容器vector中不就出现重复的答案了嘛,所以这肯定是不对的,我们最关键的就是在这里进行判断,然后实现去重,去重的操作和上述一样,然后就是一定要避免越界

      while(flag > 1 && nums[flag] == nums[flag + 1]) --flag

    • 其实到这里就差不多了,但是我们再仔细观察观察就会发现,好像flag变成负数之后,就绝对不可能有匹配的数字了,因此我们还可以对flag进行“剪枝”操作,作为第一个while循环标志你有走多少趟:
      while(flag > 1 && nums[flag] >= 0))

《四数之和》

🌹题目描述:

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

  • 0 <= a, b, c, d < n
  • abcd 互不相同
  • nums[a] + nums[b] + nums[c] + nums[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 = [2,2,2,2,2], target = 8
输出:[[2,2,2,2]]

🚗代码实现:

class Solution 
{
public:vector<vector<int>> fourSum(vector<int>& nums, int target) {sort(nums.begin(), nums.end());vector<vector<int>> vv;int index = nums.size() - 1;while(index > 2){int flag = index - 1;while(flag > 1){int left = 0, right = flag - 1;while(left < right){if(nums[left] + nums[right] > (long)target - nums[flag] - nums[index]) --right;else if(nums[left] + nums[right] < (long)target - nums[flag] - nums[index]) ++left;else{vv.push_back({nums[left], nums[right], nums[flag], nums[index]});++left, --right;while(left < right && nums[left] == nums[left - 1]) ++left;while(right > left && nums[right] == nums[right + 1])--right;}}--flag;while(flag > 1 && nums[flag] == nums[flag + 1]) --flag;}--index;while(index > 2 && nums[index] == nums[index + 1]) --index;}return vv;}
};

😴代码解析:

本道题的代码说实话就是基本照搬上面讲解的《三数之和》的代码,如果你能理解上面所讲解的《三数之和》那你一定也能理解这里的《四数之和》!

步骤一开始都是一样的,先进行排序,然后再通过双指针算法来遍历数组。这道题无疑就是多了一个数呗,那又何妨呢?思路还是一样的,只是多了一个while循环罢了。如果你理解了上述的《三数之和》,那么这道题对你来说不成问题,在这里我就不过多赘述。

值得注意的是,在处理target - nums[flag] - nums[index]时,可能会出现下列情况,这是因为出现了整形溢出的问题

nums = [1000000000,1000000000,1000000000,1000000000]

target = -29496729

Line 17: Char 71: runtime error: signed integer overflow: -1294967296 - 1000000000 cannot be represented in type 'int' (solution.cpp) SUMMARY: UndefinedBehaviorSanitizer: undefined-behavior solution.cpp:17:71

我们只需要对target - nums[flag] - nums[index]进行强制类型转换即可:(long) target - nums[flag] - nums[index.

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/53580.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

[数据集][目标检测]石油泄漏检测数据集VOC+YOLO格式6633张1类别

数据集格式&#xff1a;Pascal VOC格式YOLO格式(不包含分割路径的txt文件&#xff0c;仅仅包含jpg图片以及对应的VOC格式xml文件和yolo格式txt文件) 图片数量(jpg文件个数)&#xff1a;6633 标注数量(xml文件个数)&#xff1a;6633 标注数量(txt文件个数)&#xff1a;6633 标注…

吐血整理 ChatGPT 3.5/4.0 新手使用手册~ 【2024.09.04 更新】

以前我也是通过官网使用&#xff0c;但是经常被封号&#xff0c;就非常不方便&#xff0c;后来有朋友推荐国内工具&#xff0c;用了一阵之后&#xff0c;发现&#xff1a;稳定方便&#xff0c;用着也挺好的。 最新的 GPT-4o、4o mini&#xff0c;可搭配使用~ 1、 最新模型科普&…

【Qt】qt发布Release版本,打包.exe可执行文件

前言&#xff1a;Qt编译的可执行程序&#xff0c;如果直接运行&#xff0c;会出现0xc000007b报错&#xff0c;或者“由于占不到Qt5Network.dll,无法继续执行代码。重新安装程序可能会解决此问题”的报错&#xff0c;因为缺少相关的依赖包和动态库。 1、第一步&#xff1a;找到…

从电商行业的变化引出未来技术趋势

时间&#xff1a;2024年09月08日 作者&#xff1a;小蒋聊技术 邮箱&#xff1a;wei_wei10163.com 微信&#xff1a;wei_wei10 音频地址&#xff1a;喜马拉雅 希望大家帮个忙&#xff01;如果大家有工作机会&#xff0c;希望帮小蒋推荐一下&#xff0c;小蒋希望遇到一个认真…

Android 系统源码项目加载预编好的so库

Android 系统源码项目加载预编好的so库 文章目录 Android 系统源码项目加载预编好的so库一、前言二、源码中加载so1、Android.mk加载so加载so的主要相关代码&#xff1a; 2、Android.bp加载so&#xff08;1&#xff09;Android.mk使用源码命令编译成Android.bp&#xff08;2&am…

【C++】STL容器详解【下】

目录 一、list容器 1.1 list基本概念 1.2 lsit构造函数 1.3 list数据元素插入和删除操作 1.4 list大小操作 1.5 list赋值操作 1.6 list数据的存取 1.7 list反转排序 二、set/multiset容器 2.1 set/multiset基本概念 2.2 set构造函数 2.3 set赋值操作 2.4 set大小操…

Hash Table、HashMap、HashSet学习

文章目录 前言Hash Table&#xff08;散列表&#xff09;基本概念散列函数散列冲突&#xff08;哈希碰撞&#xff09;拉链法红黑树时间复杂度分析 HashMap基础方法使用基本的增删改查其他的方法 实现原理 HashSet基础操作去重原理 前言 本文用于介绍关于Hash Table、HashMap、…

【qt】多线程实现倒计时

1.界面设计 设置右边的intvalue从10开始倒计时 2.新建Thread类 新建Thread类&#xff0c;使其继承QThread类&#xff0c;多态重写run函数&#xff0c;相当于线程执行函数 3.重写run函数 重写run函数&#xff0c;让另一个进程每隔1s发出一个信号&#xff0c;主线程使用conne…

uniapp 全屏日历,动态无限加载

不好用请移至评论区揍我 原创代码,请勿转载,谢谢! 注:本人仅在微信小程序测试过,未在其他app/h5尝试过,按理说应该是可以的,代码没有引用任何第三方组件 日历中每个日期下方的空白部分均可自定义,写在代码中的<view class="item">我是内容</view>…

Prometheus 服务监控

目录 Prometheus 是什么 Prometheus 的特点 Prometheus 的组件 Prometheus 的指标 Prometheus 的使用场景 Prometheus 的安装 方式一&#xff1a;压缩包方式安装 方式二&#xff1a;Docker 方式安装 Prometheus 的 Web UI 面板介绍 Prometheus 目录结构介绍 Prometh…

简单的Linux Ftp服务搭建

简单的Linux FTP服务搭建 1.需求 公司有一个esb文件传输代理&#xff0c;其中我们程序有文件传输功能&#xff0c;需要将本地文件传输到esb文件代理服务器上&#xff0c;传输成功之后发送http请求&#xff0c;告知esb将固定文件进行传输到对应外围其他服务的文件目录中&#…

unity GridLayoutGroup真正的居中

GridLayoutGroup默认的居中效果: 不是真正的居中 加上代码: namespace UnityEngine.UI {/// <summary>/// GridLayoutGroup拓展&#xff0c;使支持自定义内容/// </summary>internal class GridLayoutGroupEx : GridLayoutGroup{/// <summary>/// 启用居中/…

香橙派转换模型以及在开发板上部署

新手小白记录一下自己使用香橙派模型转换以及在开发板上运行的过程&#xff0c;防止后面忘记。 使用的开发板&#xff1a;Orange Pi 5 Plus&#xff08;rk3588&#xff09; 官方的一些资料在&#xff08;主要参考用户手册&#xff09;&#xff1a;Orange Pi - Orangepihttp:/…

【视频讲解】Python贝叶斯卷积神经网络分类胸部X光图像数据集实例

全文链接&#xff1a;https://tecdat.cn/?p37604 分析师&#xff1a;Yuanchun Niu 在人工智能的诸多领域中&#xff0c;分类技术扮演着核心角色&#xff0c;其应用广泛而深远。无论是在金融风险评估、医疗诊断、安全监控还是日常的交互式服务中&#xff0c;有效的分类算法都是…

科研绘图系列:R语言富集散点图(enrichment scatter plot)

介绍 富集通路散点图(Enrichment Pathway Scatter Plot)是一种数据可视化工具,用于展示基因集富集分析(Gene Set Enrichment Analysis, GSEA)的结果。 横坐标是对应基因名称,纵坐标是通路名称,图中的点表示该基因在某个通路下的qvalue,可以简单理解为不同环境下的贡献…

【全网最全】2024年数学建模国赛A题30页完整建模文档+17页成品论文+保奖matla代码+可视化图表等(后续会更新)

您的点赞收藏是我继续更新的最大动力&#xff01; 一定要点击如下的卡片那是获取资料的入口&#xff01; 【全网最全】2024年数学建模国赛A题30页完整建模文档17页成品论文保奖matla代码可视化图表等&#xff08;后续会更新&#xff09;「首先来看看目前已有的资料&#xff0…

【ACM独立出版|EI快检索-高录用|IEEE Fellow支持】2024年数字经济与计算机科学国际学术会议(DECS2024)

【ACM独立出版&#xff5c;EI快检索-高录用&#xff5c;IEEE Fellow支持】 2024年数字经济与计算机科学国际学术会议&#xff08;DECS2024&#xff09; *ACM独立出版&#xff0c;快检索&#xff0c;高录用 *见刊后1个月左右完成EI&Scopus检索 *国内211大学、世界QS名校…

系统架构师-ERP+集成

ERP 集成平台end&#xff1a;就懒得画新的页

MyBatis-MappedStatement什么时候生成?QueryWrapper如何做到动态生成了SQL?

通过XML配置的MappedStatement 这部分MappedStatement主要是由MybatisXMLMapperBuilder进行解析&#xff0c;核心逻辑如下&#xff1a; 通过注解配置的MappedStatement 核心逻辑就在这个里面了&#xff1a; 继承BaseMapper的MappedStatement 我们看看这个类&#xff0c;里…

Java项目——苍穹外卖(一)

Entity、DTO、VO Entity&#xff08;实体&#xff09; Entity 是表示数据库表的对象&#xff0c;通常对应数据库中的一行数据。它通常包含与数据库表对应的字段&#xff0c;并可能包含一些业务逻辑。 DTO&#xff08;数据传输对象&#xff09; 作用&#xff1a;DTO 是用于在…