代码随想录 哈希表:15. 三数之和
Table of Contents
题目描述
- 给你一个整数数组 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] 输出:[[-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 。
提示:
3 <= nums.length <= 3000 -105 <= nums[i] <= 105
-
3 <= nums.length <= 3000
-
-105 <= nums[i] <= 105
思路
属于经典的双指针+排序+去重问题。
固定第一个数 nums[i],问题就变成了”在剩下的区间里找两个数,使它们的和等于 -nums[i]“。这时使用双指针很合适,前提是排序好的数组。
这其实就是”两数之和”的变种,而有序数组里找两数之和最优解法就是双指针。
题目要求不能有重复三元组。
排序后,相同的数会连在一起,跳过重复的数就能避免重复三元组。
只要遇到”找 k 个数的和为定值且不能重复”,都可以考虑”排序+双指针+去重”这个套路。
先暴力,发现超时,再思考能不能降维、利用有序性、用哈希表或双指针优化。
-
排序 首先对数组进行排序,方便后续去重和使用双指针。
-
枚举第一个数 用 for 循环枚举第一个数 nums[i],对于每个 nums[i],用双指针在剩下的区间查找另外两个数。
-
双指针查找
left 指向 i+1,right 指向数组末尾。 计算三数之和 sum = nums[i] + nums[left] + nums[right]。 如果 sum == 0,找到一个三元组,左右指针分别跳过重复元素后收缩。 如果 sum < 0,说明需要更大的数,left++。 如果 sum > 0,说明需要更小的数,right—。
- 去重
i > 0 时,若 nums[i] == nums[i-1],跳过,避免三元组重复。 找到三元组后,left/right 也要跳过重复值。
vector<vector<int>> threeSum(vector<int> &nums)
{
vector<vector<int>> result;
// 先对数组进行排序,方便后续去重和双指针查找
sort(nums.begin(), nums.end());
for (int i = 0; i < nums.size(); i++)
{
// 如果当前数字大于0,后面不可能有三数之和为0,直接结束
if (nums[i] > 0)
break;
// 跳过重复的第一个数,避免三元组重复
if (i > 0 && nums[i] == nums[i - 1])
continue;
int left = i + 1, right = nums.size() - 1;
while (left < right)
{
int sum = nums[i] + nums[left] + nums[right];
if (sum == 0)
{
result.push_back({nums[i], nums[left], nums[right]});
// 跳过重复的 left
while (left < right && nums[left] == nums[left + 1])
left++;
// 跳过重复的 right
while (left < right && nums[right] == nums[right - 1])
right--;
left++;
right--;
}
else if (sum < 0)
{
left++;
}
else
{
right--;
}
}
}
return result;
}
-
时间复杂度:O(n^2),排序 O(nlogn),主循环 O(n^2)
-
空间复杂度:O(1)(不算输出结果)
难点
- 去重技巧
i、left、right 三个指针都要注意去重,防止结果重复。 特别是 while 循环跳过重复元素的写法,容易漏掉。
- 双指针的边界处理
left < right 的条件,left/right 的移动和跳过重复的顺序。
- 提前终止
如果 nums[i] > 0,可以直接 break,提升效率。
易错点
- 去重不彻底
只对 i 去重,忘了对 left/right 去重,导致结果重复。
- 指针越界
left/right 的边界条件没处理好,可能越界或死循环。
- 排序遗漏
忘记排序,导致去重和双指针失效。
- 三元组顺序
题目要求三元组按升序排列,排序后直接按顺序 push 即可。