LeetCode第[15]题(Java):3Sum 标签:Array
题目难度:Medium
题目:
Given an array S of n integers, are there elements a, b, c in S such that a + b + c = 0? Find all unique triplets in the array which gives the sum of zero.
Note: The solution set must not contain duplicate triplets.
翻译:
给定一个n个整数的数组S,S中是否存在有a b c三个元素使得a+b+c=0?在S中找到所有的非重复组合,使它们的和为0
注意:答案集不能包含重复的数组。例如:[1,5,-6]与[-6,5,1]等数组,答案只出现其中一个
思路:三。。。个。。for循环??
(⊙x⊙;)。。。就要写!
Code:311 / 313 test cases passed. —— Time Limit Exceeded 时间复杂度:O(N3)
public static List<List<Integer>> threeSum1(int[] nums) {
Arrays.sort(nums);
List<List<Integer>> outList = new ArrayList<List<Integer>>();
for (int i = 0; i < nums.length; i++) {
if ((i > 0 && nums[i] == nums[i - 1])) // 各循环起始点不需要判断重复
continue; // 不用i++ 的原因:避免最后的k还要增加边界判断,进入下一个循环则会自动边界判断“i < nums.length”
for (int j = i + 1; j < nums.length; j++) {
if ((j > i + 1 && nums[j] == nums[j - 1]))
continue; // 当有很多for和if的时候,条件取反后用continue,以此取代if的{}缩进,使代码可读性增加
for (int k = j + 1; k < nums.length; k++) {
if ((k > j + 1 && nums[k] == nums[k - 1]))
continue;
if ((nums[i] + nums[j] + nums[k] == 0)) {
List<Integer> inList = new ArrayList<Integer>();
inList.add(nums[i]);
inList.add(nums[j]);
inList.add(nums[k]);
outList.add(inList);
break;
}
}
}
}
return outList;
}
(因为只有在尾巴进行添加并无其他功能,所以采用ArrayList比较实惠和效率)
312测试用例超时:
[82597,-9243,…………此处省略N千个数字]
不想说话,让蠢人静静。。。。
————————————————————————智商分割线————————————————————————
参考答案:74ms——beats85.12% 时间复杂度O(N2)
public List<List<Integer>> threeSum(int[] nums) {
List<List<Integer>> result = new ArrayList<List<Integer>>();
Arrays.sort(nums); for (int i = 0; i < nums.length-2; i++) {
int left = i + 1;
int right = nums.length - 1;
if (i > 0 && nums[i] == nums[i - 1])
continue; // 去掉重复的起点
while (left < right) {
int sum = nums[left] + nums[right] + nums[i];
if (sum == 0) {
result.add(Arrays.asList(new Integer[]{nums[i], nums[left], nums[right]}));
while (left < right && nums[left] == nums[left + 1])
left++; // 去掉重复的左点
while (left < right && nums[right] == nums[right - 1])
right--; // 去掉重复的右点
right--; // 进入下一组左右点判断
left++;
} else if (sum > 0) {
right--; // sum>0 ,说明和过大了,需要变小,所以移动右边指针
} else {
left++; // 同理,需要变大,移动左指针
}
}
}
return result;
}
精髓在于: (排序 + 去重) + 双指针移动相向定位
注意:不需要“重复的双胞胎数组”,所以是“组合”即Cn/m(从M个数里随机选出N个数有多少种情况)而不是排列An/m——考虑去重
所查找的数之间具有一定的关联(本题为:和为0),那么就应该利用这个属性,而不是简单的进行搜索。
思路解析:
首先考虑数组遍历时去重:方法一:先将数组排好序,在遍历的时候与上一个进行比较,相同则直接进入下一个
方法二:用容器Set——简单,但是同样需要排序,增加算法复杂度并且此题三个数操作不方便【在第一题TwoSum的后面有更新用Set的方法】。
所以数组遍历去重在已排序的情况下优先采取方法一。
其次考虑降低多次循环算法复杂度:
降低复杂度一般的途径就是利用已有的而未用到的条件将多余的步骤跳过或者删去!
由于三个数是具有一个特点的:和为某个定值,这个条件只是用来判断了而并没有使用
并且,由上个去重得知,后面使用的数组是已经排序好的。
此时仔细想想应该就能想到
从两端使用两个指针相向移动,两端指针所指数之和如果小于目标值,只需要移动左边的指针,否则只需要移动右边的指针!!
例如[1,1,1,4,5,7,8,8,9]中 定目标值为15,从两边开始,1+9为10,小于15,移动右边指针左移变成1+8只会更少,所以移动左边变成4+9以此类推。。
妈耶!我想不到啊!
此时为两数定位,而题目中要找3个数
………………外嵌for循环!!!又回到了我的领域 ahhhhhhh~
分析搞定!
LeetCode第[15]题(Java):3Sum 标签:Array的更多相关文章
- LeetCode第[15]题(Java):3Sum (三数之和为目标值)——Medium
题目难度:Medium 题目: Given an array S of n integers, are there elements a, b, c in S such that a + b + c ...
- LeetCode第[18]题(Java):4Sum 标签:Array
题目难度:Medium 题目: Given an array S of n integers, are there elements a, b, c, and d in S such that a + ...
- LeetCode第[1]题(Java):Two Sum 标签:Array
题目: Given an array of integers, return indices of the two numbers such that they add up to a specifi ...
- LeetCode第[1]题(Java):Two Sum (俩数和为目标数的下标)——EASY
题目: Given an array of integers, return indices of the two numbers such that they add up to a specifi ...
- LeetCode第[46]题(Java):Permutations(求所有全排列) 含扩展——第[47]题Permutations 2
题目:求所有全排列 难度:Medium 题目内容: Given a collection of distinct integers, return all possible permutations. ...
- LeetCode第[16]题(Java):3Sum Closest 标签:Array
题目难度:Medium 题目: Given an array S of n integers, find three integers in S such that the sum is closes ...
- LeetCode第[4]题(Java):Median of Two Sorted Arrays 标签:Array
题目难度:hard There are two sorted arrays nums1 and nums2 of size m and n respectively. Find the median ...
- LeetCode第[26]题(Java):Remove Duplicates from Sorted Array 标签:Array
题目难度:Easy 题目: Given a sorted array, remove the duplicates in-place such that each element appear onl ...
- LeetCode第[11]题(Java):Container With Most Water 标签:Array
题目难度:Medium Given n non-negative integers a1, a2, ..., an, where each represents a point at coordina ...
随机推荐
- --------------Hibernate学习(四) 多对一映射 和 一对多映射
现实中有很多场景需要用到多对一或者一对多,比如上面这两个类图所展现出来的,一般情况下,一个部门会有多名员工,一名员工只在一个部门任职. 多对一关联映射 在上面的场景中,对于Employee来说,它跟D ...
- More DETAILS! PBR的下一个发展在哪里?
最近几年图形学社区对PBR的关注非常高,也许是由于Disney以及一些游戏引擎大厂的助推,也许是因为它可以被轻松集成进实时渲染的游戏引擎当中,也许是因为许多人发现现在只需要调几个参数就能实现具有非常精 ...
- 一起读源码之zookeeper(1) -- 启动分析
从本文开始,不定期分析一个开源项目源代码,起篇从大名鼎鼎的zookeeper开始. 为什么是zk,因为用到zk的场景实在太多了,大部分耳熟能详的分布式系统都有zookeeper的影子,比如hbase, ...
- git上传到github
一. Git创建 1. git init 命令来初始化一个Git仓库 2. 添加文件到Git仓库,分两步 1) 使用命令 git add <file>,可以反复多次添加,添加多个文件 2) ...
- AJAX扩展-POST传递参数并跳转页面
拓展的代码: 这段代码的原理是创建一个表单,所有args都创建一个隐藏的input,用post方法把这些参数传递过去 注意form表单一定要加载到页面中,即下面代码中标红的部分,不然参数是无法被传递的 ...
- [转]ubuntu下安装fiddler
转 ubuntu下安装fiddler biangbiang 因为工作中需要用到fiddler工具 现在工作环境迁移到ubuntu14 下 发现fiddler只支持windows网上也有很多推荐 ...
- Android项目实战(三十九):Android集成Unity3D项目(图文详解)
需求: Unity3D 一般用于做游戏 而且是跨平台的.原本设计是Android 应用端A(原生开发)进行一些业务处理,最后由A 打开Android 应用端B(Unity3D 游戏开发)进行游戏操作. ...
- 43.Linux调试测试输入思路
当产品要发布之前,都会进行反复的测试输入,比如:测试按键,遥控,触摸等等. 当出现bug时,就还需要不停地找规律,修改程序,直到修复成功,会显的非常麻烦 答: 可以通过之前在35.Linux-分析并制 ...
- 【java】System成员输入输出功能out、in、err
package System输入输出; import java.io.ByteArrayOutputStream; import java.io.File; import java.io.FileOu ...
- iOS 设置视图背景的透明度
p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; font: 11.0px Menlo; color: #00afca } span.s1 { color: #fffff ...