LeetCode 第15题:三数之和

news/2025/2/11 21:06:33/文章来源:https://www.cnblogs.com/lavender-vv/p/18710484

LeetCode 第15题:三数之和

题目描述

给你一个整数数组 nums ,判断是否存在三元组 [nums[i], nums[j], nums[k]] 满足 i != j、i != k 且 j != k ,同时还满足 nums[i] + nums[j] + nums[k] == 0 。请你返回所有和为 0 且不重复的三元组。

注意:答案中不可以包含重复的三元组。

难度

中等

题目链接

https://leetcode.cn/problems/3sum/

示例

示例 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

解题思路

方法:排序 + 双指针

这道题是两数之和的进阶版,但直接使用两数之和的哈希表方法会导致重复,需要使用排序+双指针的方法。

关键点:

  1. 先对数组排序,便于去重和使用双指针
  2. 固定第一个数,然后在剩余部分使用双指针寻找另外两个数
  3. 注意去重:
    • 第一个数需要和前一个数不同
    • 找到答案后,左右指针都需要跳过重复值

具体步骤:

  1. 对数组进行排序
  2. 遍历数组,固定第一个数nums[i]:
    • 如果nums[i] > 0,因为数组已排序,后面不可能有三数之和为0
    • 如果nums[i]和前一个数相同,跳过以避免重复
  3. 使用双指针left和right在[i+1, n-1]范围内寻找和为-nums[i]的两个数
  4. 根据三数之和与0的关系移动指针
  5. 找到答案时注意去重

时间复杂度:O(n²)
空间复杂度:O(1)(不考虑存储答案的空间)

代码实现

C# 实现

public class Solution {public IList<IList<int>> ThreeSum(int[] nums) {List<IList<int>> result = new List<IList<int>>();if (nums == null || nums.Length < 3) {return result;}// 排序Array.Sort(nums);int n = nums.Length;// 固定第一个数for (int i = 0; i < n - 2; i++) {// 如果第一个数大于0,后面不可能有三数之和等于0if (nums[i] > 0) {break;}// 跳过重复的第一个数if (i > 0 && nums[i] == nums[i - 1]) {continue;}// 双指针寻找另外两个数int left = i + 1;int right = n - 1;while (left < right) {int sum = nums[i] + nums[left] + nums[right];if (sum == 0) {result.Add(new List<int> { nums[i], nums[left], nums[right] });// 跳过重复的左指针while (left < right && nums[left] == nums[left + 1]) {left++;}// 跳过重复的右指针while (left < right && nums[right] == nums[right - 1]) {right--;}left++;right--;}else if (sum < 0) {left++;}else {right--;}}}return result;}
}

优化版本(提前判断边界条件)

public class Solution {public IList<IList<int>> ThreeSum(int[] nums) {List<IList<int>> result = new List<IList<int>>();if (nums == null || nums.Length < 3) {return result;}Array.Sort(nums);int n = nums.Length;// 如果最小的三个数和大于0,或最大的三个数和小于0,直接返回if (nums[0] + nums[1] + nums[2] > 0 || nums[n-1] + nums[n-2] + nums[n-3] < 0) {return result;}for (int i = 0; i < n - 2; i++) {if (nums[i] > 0) break;if (i > 0 && nums[i] == nums[i - 1]) continue;// 计算当前数可能的最小和最大三数之和int minSum = nums[i] + nums[i + 1] + nums[i + 2];int maxSum = nums[i] + nums[n - 2] + nums[n - 1];if (minSum > 0) break;if (maxSum < 0) continue;int left = i + 1;int right = n - 1;while (left < right) {int sum = nums[i] + nums[left] + nums[right];if (sum == 0) {result.Add(new List<int> { nums[i], nums[left], nums[right] });while (left < right && nums[left] == nums[left + 1]) left++;while (left < right && nums[right] == nums[right - 1]) right--;left++;right--;}else if (sum < 0) {left++;}else {right--;}}}return result;}
}

代码详解

基本版本:

  1. 排序的作用:
    • 便于去重
    • 便于使用双指针
    • 可以提前结束搜索
  2. 去重处理:
    • 第一个数的去重:if (i > 0 && nums[i] == nums[i - 1])
    • 找到答案后的去重:使用while循环跳过重复值
  3. 双指针移动:
    • sum < 0 时左指针右移
    • sum > 0 时右指针左移

优化版本:

  1. 提前判断边界条件:
    • 检查最小三数之和
    • 检查最大三数之和
  2. 剪枝优化:
    • 计算当前数可能的最小和最大三数之和
    • 根据结果提前结束或跳过

执行结果

基本版本:

  • 执行用时:164 ms
  • 内存消耗:46.8 MB

优化版本:

  • 执行用时:152 ms
  • 内存消耗:46.5 MB

总结与反思

  1. 这道题的关键点:
    • 排序的重要性
    • 去重的处理
    • 双指针的应用
  2. 优化思路:
    • 提前判断边界条件
    • 计算可能的最值进行剪枝
    • 减少不必要的遍历
  3. 注意事项:
    • 数组为空或长度小于3的情况
    • 重复元素的处理
    • 整数溢出的可能

相关题目

  • LeetCode 第1题:两数之和
  • LeetCode 第16题:最接近的三数之和
  • LeetCode 第18题:四数之和
  • LeetCode 第167题:两数之和 II - 输入有序数组

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/882248.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

P1083 [NOIP 2012 提高组] 借教室(差分)

说实话竟然没想到还能这样差分,这道题我们需要二分查找m个订单,对于每次二分用一次差分,然后看如果只考虑1到mid个订单是否会出现教室不够用的情况,如果够用说明导致教室不够用的订单在后面,应该让begin=mid+1;反之让end-1;其实这道题就是让我们找第一个出现教室不够用的…

来吃糖

先放犯错后 be like:1 return // 倍增求 lca int bfs(int x) {queue<int> q;q.push(x);dep[x] = 1;while (q.size()) {x = q.front(), q.pop();for (auto it : G[x]) {int y = it.first, z = it.second;if (dep[y]) continue;dep[y] = dep[x] + 1;d[0][y] = z;f[0][y] =…

P9330 [JOISC 2023] JOI 国的节日 2 题解

Description 对于以下问题:给定长度为 \(n\) 的序列 \(a\)、\(b\),满足以下条件:在序列 \(a\) 与序列 \(b\) 中,\(1\) 到 \(2n\) 的整数各出现恰好一次; 对于 \(1\leq i\leq n\),\(a_i<b_i\); 对于 \(1\leq i<n\),\(a_i<a_{i+1}\)。求:最多能在 \([a_i,b_i]\…

一站式合同自动化:飞书审批与腾讯电子签的完美融合

Z国际教育中心专注于将全球顶尖教育资源引入中国,通过本地化整合与优化,将这些优质资源转化为中国青年触手可及的学习机会。我们的使命是帮助学生培养深厚的家国情怀与开阔的国际视野,助力他们成长为未来社会的栋梁之才。 遇到的问题 1. 业务种类多,合同审批繁琐 通过飞书平…

Linux下Docker及Nvidia Container ToolKit安装教程

作者:SkyXZ CSDN:SkyXZ~-CSDN博客 博客园:SkyXZ - 博客园 我们接下来在Ubuntu中安装Docker(安装详见:Get Docker | Docker Docs)及NVIDIA Container Toolkit(安装详见:Installing the NVIDIA Container Toolkit — NVIDIA Container Toolkit 1.17.3 documentat…

在用 uni-app 开发钉钉小程序的时候遇到一个奇怪的问题,发送请求拿不到返回的数据

今天我一位同事说用 uni-app 新开发的钉钉小程序里发送请求拿不到返回的数据,看了下发现调试工具的“Network”栏里显示请求是发送成功的,也有返回数据,但是没触发请求的回调函数。 原本用的是 luch-request 这个库发送的请求,后来试了下 uni-app 内置的 uni.request 以及钉…

15. 进程处理

一、什么是进程进程(Process)是正在运行的程序,是操作系统进行资源分配的基本单位。程序是存储在硬盘或内存的一段二进制序列,是静态的,而进程是动态的。每个进程都由自己的地址空间、代码段、数据段以及分配给它的其它系统资源(如文件描述符、网络连接等)。 二、创建子…

我把deepseek等大模型接入了微信公众号,打造个人AI助手

我把deepseek等大模型接入了微信公众号,打造个人AI助手前言 最近deepseek大模型可是火出了圈,给国产大模型公司点赞。于是乎去deepseek试了一下效果,奈何太多人使用了,问两句来一句 “服务器繁忙,请稍后再试”,体验感实在太差了。 作为程序员,怎么能忍受?于是乎去寻找d…

(未解决)word中插入pdf图片(高清、矢量图)

(未解决)word中插入pdf图片(高清、矢量图) 1.个人诉求: pdf高清矢量格式的图片,插入至word中 2.尝试的解决方法: https://blog.csdn.net/weixin_45399376/article/details/115281547?spm=1001.2014.3001.5502。 3.具体操作步骤: 1)word——插入——对象——文件中的文…

一语总结

一语总结 真心建议把找到的一些性质和做法以文字形式写下来,当掉不过样例的时候一个一个检查其正确性。”P9169 [省选联考 2023] 过河卒“用DFS将环当作平局是错误的因为这可能只是一个不优的必胜/必败局面,正确的做法是老实建图按拓扑序跑BFS。就这个结论的错误害我虚空调题…

软考高级《系统架构设计师》知识点(二)

操作系统知识 操作系统概述操作系统定义:能有效地组织和管理系统中的各种软/硬件资源,合理地组织计算机系统工作流程,控制程序的执行,并且向用户提供一个良好的工作环境和友好的接口。 操作系统有三个重要的作用:管理计算机中运行的程序和分配各种软硬件资源; 为用户提供…