https://leetcode-cn.com/problems/3sum/submissions/
最优解
class Solution {public List<List<Integer>> threeSum(int[] nums) {Arrays.sort(nums);List<List<Integer>> ans = new ArrayList<>();for (int i = 0; i < nums.length - 2; i++) {if (nums[i] > 0) {return ans;}if (i > 0 && nums[i] == nums[i - 1]) {continue;}int left = i + 1, right = nums.length - 1;while (left < right) {int sum = nums[i] + nums[left] + nums[right];if (sum == 0) {ans.add(Arrays.asList(nums[i], nums[left], nums[right]));left++;right--;while (left < right && nums[left] == nums[left - 1]) {left++;}while (left < right && nums[right] == nums[right + 1]) {right--;}} else if (sum > 0) {right--;} else {left++;}}}return ans;}}
先找一个目标值nums[i],再用双指针扫描剩余。把三数之和转化为两数之和,这里用双指针扫描而不是用两数之和的map方法,原因也是随着目标指针i的右移,剩余长度越来越短,扫描时间可控,同时如果使用map,value也是列表结构,也许进行遍历。
这里还有个疑问,在sum > 0 或 < 0时也可以用while循环去掉重复元素,理论上也不会更慢,但实际计算耗时增加很多。
如果使用双指针+二分查找方式的问题是当sum=0是不确定左右指针的移动方向,需要递归计算。
