题目分析
题目链接:15. 3Sum
此题要求从数组中找出和为0的三元组,实际上这个“0”可以换为任意一个数字,算法都是一样的。
搜索问题的思路
这又是一个搜索的问题(从数组中搜索满足条件的三元组),我在另一篇博客中总结过搜索问题的一般解法:
搜索最佳结果的一个一般思想:先想象这个最佳结果必定具有什么样的特征,然后我们通过这些特征来查找,不断找出很多“候选最佳结果”,最后从这些候选结果中选拔出最佳的结果。
那么,和为0的三元组满足什么条件呢?首先,不妨让三元组的3个元素从小到大排列,因此第一个元素就是最小的元素。如果第一个元素为left,则另外两个元素之和为-left,这是一个充要条件。
因此,搜索的思路就出来了:依次对每个数字都假定它是left,然后从大于等于left的数字中找到和为-left的两个数字,如果找到了,则这三个数字可以组成和为0的三元组。
为什么只需要从大于等于left的数字中找?因为我们已经假设了三元组按照从小到达排列。这个假设不妨碍我们找到所有符合条件的三元组,而且大大降低了搜索的时间。
算法步骤
- 首先对输入数组
vector<int>& nums
进行排序。这不但有利于我们依次对每个数字都假定它是left,而且有利于我们从大于等于left的数字中找到和为-left的两个数字。 - 从小到大,依次假设每一个数字为left。
- 从数组的
left_index+1
~size()-1
下标范围内,找到2个和为-left的数字。如果找到了就将三元组放入结果集合vector<vector<int>> res
中,然后在这个下标范围继续搜索。 - 数组的
left_index+1
~size()-1
下标范围搜索完毕后,回到步骤2,假设下一个数字为left。直到所有数字都已经被假设过为left。 - 返回结果集合
res
。
代码实现
class Solution {
public:
vector<vector<int>> threeSum(vector<int>& nums) {
vector<vector<int>> res;
int length = nums.size();
std::sort(nums.begin(), nums.end());
for (int left = 0; left < length; left++) {
int shouldHaveSum = -1*nums[left];
int i = left+1, j = length-1;
while (i < j) {
int sum = nums[i]+nums[j];
if (sum == shouldHaveSum) {
vector<int> triplet(3);
triplet[0] = nums[left];
triplet[1] = nums[i];
triplet[2] = nums[j];
res.push_back(triplet);
// 为了不重复输出三元组,我们跳过已经在上一个三元组中的数字
while (i < j && nums[i] == triplet[1]) ++i;
while (i < j && nums[j] == triplet[2]) --j;
}
else if (sum < shouldHaveSum) {
// sum太小,通过增加第二个数字让sum大一些
++i;
}
else if (sum > shouldHaveSum) {
// sum太大,通过减小第三个数字让sum小一些
--j;
}
}
// 为了不重复输出三元组,我们不多次假设同一个数字为left
while (left+1 < length && nums[left+1] == nums[left]) ++left;
}
return res;
}
};
时间复杂度
最外层循环(依次假设每个数字为left)做length次。内层循环相当于遍历那些在left右边的所有数字,时间复杂度为O(length)。
因此总的时间复杂度为length*O(length),也就是O(length^2)。
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。