剑指 Offer II 007. 数组中和为 0 的三个数


comments: true
edit_url: https://github.com/doocs/leetcode/edit/main/lcof2/%E5%89%91%E6%8C%87%20Offer%20II%20007.%20%E6%95%B0%E7%BB%84%E4%B8%AD%E5%92%8C%E4%B8%BA%200%20%E7%9A%84%E4%B8%89%E4%B8%AA%E6%95%B0/README.md

剑指 Offer II 007. 数组中和为 0 的三个数

题目描述

给定一个包含 n 个整数的数组 nums,判断 nums 中是否存在三个元素 abc 使得 a + b + c = 0 ?请找出所有和为 0 且 不重复 的三元组。

 

示例 1:

输入:nums = [-1,0,1,2,-1,-4]
输出:[[-1,-1,2],[-1,0,1]]

示例 2:

输入:nums = []
输出:[]

示例 3:

输入:nums = [0]
输出:[]

 

提示:

  • 0 <= nums.length <= 3000
  • -105 <= nums[i] <= 105

 

注意:本题与主站 15 题相同:https://leetcode.cn/problems/3sum/

解法

方法一:排序 + 双指针

我们注意到,题目不要求我们按照顺序返回三元组,因此我们不妨先对数组进行排序,这样就可以方便地跳过重复的元素。

接下来,我们枚举三元组的第一个元素 n u m s [ i ] nums[i] nums[i],其中 0 ≤ i < n − 2 0 \leq i \lt n - 2 0i<n2。对于每个 i i i,我们可以通过维护两个指针 j = i + 1 j = i + 1 j=i+1 k = n − 1 k = n - 1 k=n1,从而找到满足 n u m s [ i ] + n u m s [ j ] + n u m s [ k ] = 0 nums[i] + nums[j] + nums[k] = 0 nums[i]+nums[j]+nums[k]=0 j j j k k k。在枚举的过程中,我们需要跳过重复的元素,以避免出现重复的三元组。

具体判断逻辑如下:

如果 i > 0 i \gt 0 i>0 并且 n u m s [ i ] = n u m s [ i − 1 ] nums[i] = nums[i - 1] nums[i]=nums[i1],则说明当前枚举的元素与上一个元素相同,我们可以直接跳过,因为不会产生新的结果。

如果 n u m s [ i ] > 0 nums[i] \gt 0 nums[i]>0,则说明当前枚举的元素大于 0 0 0,则三数之和必然无法等于 0 0 0,结束枚举。

否则,我们令左指针 j = i + 1 j = i + 1 j=i+1,右指针 k = n − 1 k = n - 1 k=n1,当 j < k j \lt k j<k 时,执行循环,计算三数之和 x = n u m s [ i ] + n u m s [ j ] + n u m s [ k ] x = nums[i] + nums[j] + nums[k] x=nums[i]+nums[j]+nums[k],并与 0 0 0 比较:

  • 如果 x < 0 x \lt 0 x<0,则说明 n u m s [ j ] nums[j] nums[j] 太小,我们需要将 j j j 右移一位。
  • 如果 x > 0 x \gt 0 x>0,则说明 n u m s [ k ] nums[k] nums[k] 太大,我们需要将 k k k 左移一位。
  • 否则,说明我们找到了一个合法的三元组,将其加入答案,并将 j j j 右移一位,将 k k k 左移一位,同时跳过所有重复的元素,继续寻找下一个合法的三元组。

枚举结束后,我们即可得到三元组的答案。

时间复杂度 O ( n 2 ) O(n^2) O(n2),空间复杂度 O ( log ⁡ n ) O(\log n) O(logn)。其中 n n n 为数组的长度。

Python3
class Solution:def threeSum(self, nums: List[int]) -> List[List[int]]:nums.sort()res=[]for i in range(len(nums)):if i and nums[i]==nums[i-1]:continue#两数之和:o(n)j,k=i+1,len(nums)-1while j<k:x=nums[i]+nums[j]+nums[k]if x>0:k-=1elif x<0:j+=1else:res.append([nums[i],nums[j],nums[k]])#更新j,k, 继续找下一组while j<k and nums[j]==nums[j+1]:j+=1 #nums[j]!=nums[j+1]结束,所以更新j=j+1while j<k and nums[k]==nums[k-1]:k-=1 #nums[k]!=nums[k-1]结束,所以更新k=k-1j=j+1k=k-1return res
Java
class Solution {public List<List<Integer>> threeSum(int[] nums) {Arrays.sort(nums);List<List<Integer>> ans = new ArrayList<>();int n = nums.length;for (int i = 0; i < n - 2 && nums[i] <= 0; ++i) {if (i > 0 && nums[i] == nums[i - 1]) {continue;}int j = i + 1, k = n - 1;while (j < k) {int x = nums[i] + nums[j] + nums[k];if (x < 0) {++j;} else if (x > 0) {--k;} else {ans.add(List.of(nums[i], nums[j++], nums[k--]));while (j < k && nums[j] == nums[j - 1]) {++j;}while (j < k && nums[k] == nums[k + 1]) {--k;}}}}return ans;}
}
C++
class Solution {
public:vector<vector<int>> threeSum(vector<int>& nums) {sort(nums.begin(), nums.end());vector<vector<int>> ans;int n = nums.size();for (int i = 0; i < n - 2 && nums[i] <= 0; ++i) {if (i && nums[i] == nums[i - 1]) {continue;}int j = i + 1, k = n - 1;while (j < k) {int x = nums[i] + nums[j] + nums[k];if (x < 0) {++j;} else if (x > 0) {--k;} else {ans.push_back({nums[i], nums[j++], nums[k--]});while (j < k && nums[j] == nums[j - 1]) {++j;}while (j < k && nums[k] == nums[k + 1]) {--k;}}}}return ans;}
};
Go
func threeSum(nums []int) (ans [][]int) {sort.Ints(nums)n := len(nums)for i := 0; i < n-2 && nums[i] <= 0; i++ {if i > 0 && nums[i] == nums[i-1] {continue}j, k := i+1, n-1for j < k {x := nums[i] + nums[j] + nums[k]if x < 0 {j++} else if x > 0 {k--} else {ans = append(ans, []int{nums[i], nums[j], nums[k]})j, k = j+1, k-1for j < k && nums[j] == nums[j-1] {j++}for j < k && nums[k] == nums[k+1] {k--}}}}return
}
TypeScript
function threeSum(nums: number[]): number[][] {nums.sort((a, b) => a - b);const ans: number[][] = [];const n = nums.length;for (let i = 0; i < n - 2 && nums[i] <= 0; i++) {if (i > 0 && nums[i] === nums[i - 1]) {continue;}let j = i + 1;let k = n - 1;while (j < k) {const x = nums[i] + nums[j] + nums[k];if (x < 0) {++j;} else if (x > 0) {--k;} else {ans.push([nums[i], nums[j++], nums[k--]]);while (j < k && nums[j] === nums[j - 1]) {++j;}while (j < k && nums[k] === nums[k + 1]) {--k;}}}}return ans;
}
Rust
use std::cmp::Ordering;impl Solution {pub fn three_sum(mut nums: Vec<i32>) -> Vec<Vec<i32>> {nums.sort();let n = nums.len();let mut res = vec![];let mut i = 0;while i < n - 2 && nums[i] <= 0 {let mut l = i + 1;let mut r = n - 1;while l < r {match (nums[i] + nums[l] + nums[r]).cmp(&0) {Ordering::Less => {l += 1;}Ordering::Greater => {r -= 1;}Ordering::Equal => {res.push(vec![nums[i], nums[l], nums[r]]);l += 1;r -= 1;while l < n && nums[l] == nums[l - 1] {l += 1;}while r > 0 && nums[r] == nums[r + 1] {r -= 1;}}}}i += 1;while i < n - 2 && nums[i] == nums[i - 1] {i += 1;}}res}
}
JavaScript
/*** @param {number[]} nums* @return {number[][]}*/
var threeSum = function (nums) {const n = nums.length;nums.sort((a, b) => a - b);const ans = [];for (let i = 0; i < n - 2 && nums[i] <= 0; ++i) {if (i > 0 && nums[i] === nums[i - 1]) {continue;}let j = i + 1;let k = n - 1;while (j < k) {const x = nums[i] + nums[j] + nums[k];if (x < 0) {++j;} else if (x > 0) {--k;} else {ans.push([nums[i], nums[j++], nums[k--]]);while (j < k && nums[j] === nums[j - 1]) {++j;}while (j < k && nums[k] === nums[k + 1]) {--k;}}}}return ans;
};
C#
public class Solution {public IList<IList<int>> ThreeSum(int[] nums) {Array.Sort(nums);int n = nums.Length;IList<IList<int>> ans = new List<IList<int>>();for (int i = 0; i < n - 2 && nums[i] <= 0; ++i) {if (i > 0 && nums[i] == nums[i - 1]) {continue;}int j = i + 1, k = n - 1;while (j < k) {int x = nums[i] + nums[j] + nums[k];if (x < 0) {++j;} else if (x > 0) {--k;} else {ans.Add(new List<int> { nums[i], nums[j--], nums[k--] });while (j < k && nums[j] == nums[j + 1]) {++j;}while (j < k && nums[k] == nums[k + 1]) {--k;}}}}return ans;}
}
Ruby
# @param {Integer[]} nums
# @return {Integer[][]}
def three_sum(nums)res = []nums.sort!for i in 0..(nums.length - 3)next if i > 0 && nums[i - 1] == nums[i]j = i + 1k = nums.length - 1while j < k dosum = nums[i] + nums[j] + nums[k]if sum < 0j += 1elsif sum > 0k -= 1elseres += [[nums[i], nums[j], nums[k]]]j += 1k -= 1j += 1 while nums[j] == nums[j - 1]k -= 1 while nums[k] == nums[k + 1]endendendres
end
Swift
class Solution {func threeSum(_ nums: [Int]) -> [[Int]] {if nums.count < 3 {return []}let nums = nums.sorted()var ans = [[Int]]()let n = nums.countfor i in 0..<n-2 {if nums[i] > 0 { break }if i > 0 && nums[i] == nums[i - 1] { continue }var j = i + 1var k = n - 1while j < k {let sum = nums[i] + nums[j] + nums[k]if sum < 0 {j += 1} else if sum > 0 {k -= 1} else {ans.append([nums[i], nums[j], nums[k]])j += 1k -= 1while j < k && nums[j] == nums[j - 1] { j += 1 }while j < k && nums[k] == nums[k + 1] { k -= 1 }}}}return ans}
}

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

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

相关文章

2025春晚刘谦魔术揭秘魔术过程

2025春晚刘谦魔术揭秘魔术过程 首先来看全过程 将杯子&#xff0c;筷子&#xff0c;勺子以任意顺序摆成一排 1.筷子和左边物体交换位置 2.杯子和右边物体交换位置 3.勺子和左边物体交换位置 最终魔术的结果是右手出现了杯子 这个就是一个简单的分类讨论的问题。 今年的魔术…

具身智能研究报告

参考&#xff1a; &#xff08;1&#xff09;GTC大会&Figure&#xff1a;“具身智能”奇点已至 &#xff08;2&#xff09;2024中国具身智能创投报告 &#xff08;3&#xff09;2024年具身智能产业发展研究报告 &#xff08;4&#xff09;具身智能行业深度&#xff1a;发展…

将ollama迁移到其他盘(eg:F盘)

文章目录 1.迁移ollama的安装目录2.修改环境变量3.验证 背景&#xff1a;在windows操作系统中进行操作 相关阅读 &#xff1a;本地部署deepseek模型步骤 1.迁移ollama的安装目录 因为ollama默认安装在C盘&#xff0c;所以只能安装好之后再进行手动迁移位置。 # 1.迁移Ollama可…

VPR概述、资源

SOTA网站&#xff1a; Visual Place Recognition | Papers With Code VPR&#xff08;Visual Place Recognition&#xff09; 是计算机视觉领域的一项关键任务&#xff0c;旨在通过图像匹配和分析来识别场景或位置。它的目标是根据视觉信息判断某个场景是否与数据库中的场景匹…

(Java版本)基于JAVA的网络通讯系统设计与实现-毕业设计

源码 论文 下载地址&#xff1a; ​​​​c​​​​​​c基于JAVA的网络通讯系统设计与实现(源码系统论文&#xff09;https://download.csdn.net/download/weixin_39682092/90299782https://download.csdn.net/download/weixin_39682092/90299782 第1章 绪论 1.1 课题选择的…

MySQL中的读锁与写锁:概念与作用深度剖析

MySQL中的读锁与写锁&#xff1a;概念与作用深度剖析 在MySQL数据库的并发控制机制中&#xff0c;读锁和写锁起着至关重要的作用。它们是确保数据在多用户环境下能够正确、安全地被访问和修改的关键工具。 一、读锁&#xff08;共享锁&#xff09;概念 读锁&#xff0c;也称为…

用HTML、CSS和JavaScript实现庆祝2025蛇年大吉(附源码)

用HTML、CSS和JavaScript庆祝2025蛇年大吉 在这个数字化时代&#xff0c;网页设计不仅仅是为了展示信息&#xff0c;更是传达情感和文化的一种方式。2025年将是蛇年&#xff0c;许多人希望通过各种方式庆祝这一重要的时刻。在这篇文章中&#xff0c;我们将一起学习如何使用HTM…

STM32标准库移植RT-Thread nano

STM32标准库移植RT-Thread Nano 哔哩哔哩教程链接&#xff1a;STM32F1标准库移植RT_Thread Nano 移植前的准备 stm32标准库的裸机代码&#xff08;最好带有点灯和串口&#xff09;RT-Thread Nano Pack自己的开发板 移植前的说明 本人是在读学生&#xff0c;正在学习阶段&a…

Unity 粒子特效在UI中使用裁剪效果

1.使用Sprite Mask 首先建立一个粒子特效在UI中显示 新建一个在场景下新建一个空物体&#xff0c;添加Sprite Mask组件&#xff0c;将其的Layer设置为UI相机渲染的UI层&#xff0c; 并将其添加到Canvas子物体中&#xff0c;调整好大小&#xff0c;并选择合适的Sprite&#xff…

JVM栈溢出线上环境排查

#查看当前Linux系统进程ID、线程ID、CPU占用率&#xff08;-eo后面跟想要展示的列&#xff09; ps H -eo pid,tid,%cpups H -eo pid,tid,%cpu |grep tid #使用java jstack 查看进程id下所有线程id的情况 jstack pid 案例2 通过jstack 排查死锁问题 #启动java代码 jstack 进…

【Linux权限】—— 于虚拟殿堂,轻拨密钥启华章

欢迎来到ZyyOvO的博客✨&#xff0c;一个关于探索技术的角落&#xff0c;记录学习的点滴&#x1f4d6;&#xff0c;分享实用的技巧&#x1f6e0;️&#xff0c;偶尔还有一些奇思妙想&#x1f4a1; 本文由ZyyOvO原创✍️&#xff0c;感谢支持❤️&#xff01;请尊重原创&#x1…

(2)SpringBoot自动装配原理简介

SpringBoot自动装配 这里写目录标题 SpringBoot自动装配启动器主程序自定义扫描包SpringBootApplicationSpringBootConfigurationEnableAutoConfigurationAutoConfigurationPackageImport({AutoConfigurationImportSelector.class})选择器AutoConfigurationEntrygetCandidateCo…

计算机网络 (60)蜂窝移动通信网

一、定义与原理 蜂窝移动通信网是指将一个服务区分为若干蜂窝状相邻小区并采用频率空间复用技术的移动通信网。其原理在于&#xff0c;将移动通信服务区划分成许多以正六边形为基本几何图形的覆盖区域&#xff0c;称为蜂窝小区。每个小区设置一个基站&#xff0c;负责本小区内移…

17.Word:李楠-学术期刊❗【29】

目录 题目​ NO1.2.3.4.5 NO6.7.8 NO9.10.11 NO12.13.14.15 NO16 题目 NO1.2.3.4.5 另存为手动/F12Fn光标来到开头位置处→插入→封面→选择花丝→根据样例图片&#xff0c;对应位置填入对应文字 (手动调整即可&#xff09;复制样式&#xff1a;开始→样式对话框→管理…

Java面试题2025-并发编程基础(多线程、锁、阻塞队列)

并发编程 一、线程的基础概念 一、基础概念 1.1 进程与线程A 什么是进程&#xff1f; 进程是指运行中的程序。 比如我们使用钉钉&#xff0c;浏览器&#xff0c;需要启动这个程序&#xff0c;操作系统会给这个程序分配一定的资源&#xff08;占用内存资源&#xff09;。 …

Java创建项目准备工作

新建项目 新建空项目 每一个空项目创建好后都要检查jdk版本 检查SDK和语言级别——Apply——OK 检查当前项目的Maven路径&#xff0c;如果已经配置好全局&#xff0c;就是正确路径不用管 修改项目字符集编码&#xff0c;将所有编码都调整为UTF-8 创建Spingboot工程 创建Spring…

2007-2020年各省国内专利申请授权量数据

2007-2020年各省国内专利申请授权量数据 1、时间&#xff1a;2007-2020年 2、来源&#xff1a;国家统计局、统计年鉴 3、指标&#xff1a;行政区划代码、地区名称、年份、国内专利申请授权量(项) 4、范围&#xff1a;31省 5、指标解释&#xff1a;专利是专利权的简称&…

(一)QT的简介与环境配置WIN11

目录 一、QT的概述 二、QT的下载 三、简单编程 常用快捷键 一、QT的概述 简介 Qt&#xff08;发音&#xff1a;[kjuːt]&#xff0c;类似“cute”&#xff09;是一个跨平台的开发库&#xff0c;主要用于开发图形用户界面&#xff08;GUI&#xff09;应用程序&#xff0c;…

【C语言】main函数解析

一、前言 在学习编程的过程中&#xff0c;我们很早就接触到了main函数。在Linux系统中&#xff0c;当你运行一个可执行文件&#xff08;例如 ./a.out&#xff09;时&#xff0c;如果需要传入参数&#xff0c;就需要了解main函数的用法。本文将详细解析main函数的参数&#xff…

自创《艺术人生》浅析

艺术是生活的馈赠&#xff0c;艺术是苦痛的呻吟。 笔记模板由python脚本于2025-01-29 00:01:11创建&#xff0c;本篇笔记适合喜欢写诗读诗诵诗的coder翻阅。 【学习的细节是欢悦的历程】 博客的核心价值&#xff1a;在于输出思考与经验&#xff0c;而不仅仅是知识的简单复述。 …