力扣15(Java)-三数之和(中等)
题目:
给你一个整数数组 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
来源:力扣(LeetCode)
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。
解题思路:
双指针
参考:代码随想录的文字讲解 && 代码随想录的视频讲解
- 需要求的是数字而不是索引,可以先将数组排序,利用一层for循环,从下标0开始遍历数组元素,即nums[i],再定义双指针从left = i+1,right = nums.length - 1开始往中间靠拢,寻找sum = nums[i] + nums[left] + nums[right]
- 如果sum > 0, 说明和大了,应该减小,则移动右指针使其数字变小,即right--;
- 如果sum < 0,说明和小了,应该增大,则移动左指针使其数字变大,即left++;
- 如果sum == 0, 说明找到了一个符合题意的三元组,即加入结果集。
细节问题:
①答案中不应该包含重复的三元组[三元组内的数字可以重复],如果nums[i]有重复的应该在怎么去重,是判断nums[i] == nums[i+1] 还是nums[i] == nums[i-1]的时候跳过当前数字呢?
例如:示例1中的nums = [-1,0,1,2,-1,-4],排序后nums = [-4,-1,-1,0,1,2]
- 如果判断 nums[i] == nums[i+1] 的时候跳过:nums[1] == nums[2],有[-1,-1,2]和[-1,0,1]这两个符合要求的三元组,这时候跳过,下一步遍历到nums[2]== -1时,就只有[-1,0,1],总体就会跳过[-1,-1,2这个符合要求的三元组。
- 如果判断 nums[i] == nums[i-1] 的时候跳过:遍历到nums[1] == -1的时候有[-1,-1,2]和[-1,0,1]这两个符合要求的三元组,下一步遍历到nums[2]== -1时,nums[2] == nums[1]时,就有[-1,0,1]这时候与nums[1]得到的三元组重复了,就需要跳过了。
②左右指针的去重应该放在找到第一个三元组之后去重:例如:[0,0,0,0,0]
如果按照下面这样写,那么就不会得到任何一个满足条件的三元组,但是实际上是有一个符合条件的三元组的[0,0,0],所以bc的去重应该写在得到第一个符合要求的三元组后面。
while (left < right){
//如果将bc去重逻辑放在while后面
while (left < right && nums[left] == nums[left + 1]) left++;
while (left < right && nums[right] == nums[right - 1]) right--;
.....
}
代码:
1 class Solution {
2 public List<List<Integer>> threeSum(int[] nums) {
3 ArrayList<List<Integer>> ans = new ArrayList<>();
4 Arrays.sort(nums);
5 for (int i = 0; i < nums.length; i++){
6 //如果第一个数(最小的数)都大于0,后面相加只会更大不会等于0
7 if (nums[i] > 0) return ans;
8 //去重,避免重复的三元组
9 if (i > 0 && nums[i] == nums[i-1]) continue;
10 int left = i + 1, right = nums.length - 1;
11 while (left < right){
12 int sum = nums[i] + nums[left] + nums[right];
13 //如果和大于0,让和减小移动右指针
14 if (sum > 0){
15 right--;
16 }else if (sum < 0){
17 //和小于0,让和变大,移动左指针
18 left++;
19 }else{
20 //和等于0,找到一个三元组添加进结果
21 ans.add(Arrays.asList(nums[i], nums[left], nums[right]));
22 //去除左右指针后面和前面的重复元素,例如[-1,-1,-1,0,1,1,1]
23 while (left < right && nums[left] == nums[left + 1]) left++;
24 while (left < right && nums[right] == nums[right - 1]) right--;
25 left++;
26 right--;
27 }
28 }
29 }
30 return ans;
31 }
32 }
小知识:
Arrays.asList(): 该方法是将数组转化成List集合的方法。
- 用此方法得到的List的长度是不可改变的;
- 不支持add()、remove()、clear()等方法;
- 将数组转换为ArrayList:List list = new ArrayList<>(Arrays.asList("a", "b", "c"))
力扣15(Java)-三数之和(中等)的更多相关文章
- 【LeetCode】15. 3Sum 三数之和
作者: 负雪明烛 id: fuxuemingzhu 个人博客:http://fuxuemingzhu.cn/ 个人公众号:负雪明烛 本文关键词:3sum, 三数之和,题解,leetcode, 力扣,P ...
- [LeetCode] 15. 3Sum 三数之和
Given an array S of n integers, are there elements a, b, c in S such that a + b + c = 0? Find all un ...
- LeeCode数组第15题三数之和
题目:三数之和 内容: 给定一个包含 n 个整数的数组 nums,判断 nums 中是否存在三个元素 a,b,c ,使得 a + b + c = 0 ?找出所有满足条件且不重复的三元组. 注意:答案中 ...
- 【LeetCode】15、三数之和为0
题目等级:3Sum(Medium) 题目描述: Given an array nums of n integers, are there elements a, b, c in nums such t ...
- LeetCode 第15题-三数之和
1. 题目 2.题目分析与思路 3.思路 1. 题目 给定一个包含 n 个整数的数组 nums,判断 nums 中是否存在三个元素 a,b,c ,使得 a + b + c = 0 ?找出所有满足条件且 ...
- [leetcode]15. 3Sum三数之和
Given an array nums of n integers, are there elements a, b, c in nums such that a + b + c = 0? Find ...
- 力扣 ——4Sum (四数之和)python 实现
题目描述: 中文: 给定一个包含 n 个整数的数组 nums 和一个目标值 target,判断 nums 中是否存在四个元素 a,b,c 和 d ,使得 a + b + c + d 的值与 targe ...
- Leetcode(15)-三数之和
给定一个包含 n 个整数的数组 nums,判断 nums 中是否存在三个元素 a,b,c ,使得 a + b + c = 0 ?找出所有满足条件且不重复的三元组. 注意:答案中不可以包含重复的三元组. ...
- leetcode 刷题(数组篇)15题 三数之和 (双指针)
很有意思的一道题,值得好好思考,虽然难度只有Mid,但是个人觉得不比Hard简单 题目描述 给你一个包含 n 个整数的数组 nums,判断 nums 中是否存在三个元素 a,b,c ,使得 a + b ...
- 【LeetCode 15】三数之和
题目链接 [题解] 先把n个数字升序排个序. 然后枚举三元组最左边的那个数字是第i个数字. 之后用两个指针l,r移动来获取三元组的第2个和第3个数字. (初始值,l=i+1,r = n-1); 如果a ...
随机推荐
- 【大语言模型基础】-详解Transformer原理
一.Transformer Transformer最开始用于机器翻译任务,其架构是seq2seq的编码器解码器架构.其核心是自注意力机制: 每个输入都可以看到全局信息,从而缓解RNN的长期依赖问题. ...
- Apollo3 Blue SoC 和 Apollo3 Blue Plus SoC的区别
一 芯片简介 1.简介 Apollo3 Blue系列SoC解决方案代表了超低功耗设计的巨大飞跃,其运行模式和睡眠模式下的功耗数值均十分出色,并且具有高性能的处理引擎. Apollo3Blue器件提供了 ...
- C语言中的强制转换
许久没有遇到的问题 C语言真是博大精深,越使用它,就越发感觉到它的威力和恐怖,最近在做算法的时候,遇到了一个强转的错误,把人折腾的够受,这次要好好梳理一下了,希望下次不能再犯此类的问题. 强制转换 ...
- 【leetcode 春季比赛3题 二叉搜索树染色】广度搜索
暴力: import java.util.ArrayList; import java.util.Collections; import java.util.Comparator; import ja ...
- 3DCAT实时云渲染助力广府庙会元宇宙焕新亮相,开启线上奇趣之旅!
超 400 万人次打卡,商圈营业额逾 3.6 亿元,2023 年广府庙会于2023年2月11日圆满落幕. 活动期间,佳境美如画,融合VR.AR.虚拟直播等技术的广府庙会元宇宙焕新亮相,群众只需点击一个 ...
- 前端 Typescript 入门
前端 Typescript 入门 Ant design vue4.x 基于 vue3,示例默认是 TypeScript.比如 table 组件管理. vue3 官网介绍也使用了 TypeScript, ...
- 采用DevOps的7个主要障碍,你一定不知道!
尽管DevOps已经相对成熟,DevOps哲学仍然在回避甚至是最著名和最有资源的组织.一份令人震惊的Gartner报告显示,75%的DevOps项目未能实现其目标.为什么DevOps的失败率如此之高? ...
- #min-max容斥#51nod 1355 斐波那契的最小公倍数
题目 对于 \(n\leq 50000,a_i\leq 10^6\),求 \(\large lcm(fib(a_1),fib(a_2),\dots,fib(a_{n-1}),fib(a_n))\) 分 ...
- #矩阵树定理,高斯消元,容斥定理#洛谷 4336 [SHOI2016]黑暗前的幻想乡
题目 分析 这很明显是矩阵树定理,但是每个建筑公司都恰好修建一条边非常难做, 考虑如果一个建筑公司在某个方案中并没有恰好修建一条边, 那么这种方案一定能在不选其它任意一个公司的方案中被减掉, 那就可以 ...
- 【译】如何在 Visual Studio 中安装 GitHub Copilot
GitHub Copilot 简介 GitHub Copilot 是一个新工具,可以帮助您在人工智能的帮助下更快,更智能地编写代码.它可以建议代码补全,生成代码片段,甚至为您编写整个函数.GitHub ...