LeetCodeAnimation/notes/LeetCode第15号问题:三数之和.md

58 lines
2.2 KiB
Java
Raw Normal View History

2019-05-02 15:59:01 +08:00
# LeetCode 15 号问题三数之和
> 本文首发于公众号五分钟学算法[图解 LeetCode ](<https://github.com/MisterBooo/LeetCodeAnimation>)系列文章之一。
>
> 个人网站[https://www.cxyxiaowu.com](https://www.cxyxiaowu.com)
题目来源于 LeetCode 上第 15 号问题三数之和
### 题目描述
给定一个包含 *n* 个整数的数组 `nums`判断 `nums` 中是否存在三个元素 *abc *使得 *a + b + c =* 0 找出所有满足条件且不重复的三元组
### 题目解析
题目需要我们找出三个数且和为 0 那么除了三个数全是 0 的情况之外肯定会有负数和正数所以一开始可以先选择一个数然后再去找另外两个数这样只要找到两个数且和为第一个选择的数的相反数就行了也就是说需要枚举 a b c 的存入 map 即可
需要注意的是返回的结果中不能有有重复的结果这样的代码时间复杂度是 O(n^2)在这里可以先将原数组进行排序然后再遍历排序后的数组这样就可以使用双指针以线性时间复杂度来遍历所有满足题意的两个数组合
### 动画描述
待补充
### 代码实现
###
```c++
class Solution {
public:
vector<vector<int>> threeSum(vector<int>& nums) {
vector<vector<int>> res;
sort(nums.begin(), nums.end());
if (nums.empty() || nums.back() < 0 || nums.front() > 0) return {};
for (int k = 0; k < nums.size(); ++k) {
if (nums[k] > 0) break;
if (k > 0 && nums[k] == nums[k - 1]) continue;
int target = 0 - nums[k];
int i = k + 1, j = nums.size() - 1;
while (i < j) {
if (nums[i] + nums[j] == target) {
res.push_back({nums[k], nums[i], nums[j]});
while (i < j && nums[i] == nums[i + 1]) ++i;
while (i < j && nums[j] == nums[j - 1]) --j;
++i; --j;
} else if (nums[i] + nums[j] < target) ++i;
else --j;
}
}
return res;
}
};
```
![](https://bucket-1257126549.cos.ap-guangzhou.myqcloud.com/blog/fz0rq.png)