【每日一题】LeetCode 15. 三数之和 TypeScript

给你一个整数数组 nums ,判断是否存在三元组 [nums[i], nums[j], nums[k]] 满足 i != ji != kj != 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

TypeScript 复制代码
function threeSum(nums: number[]): number[][] {
    const three:number[][] = []
    nums.sort((a,b)=>a-b)
    for(let i=0;i<nums.length-2;i++){
        if(i>0 && nums[i]===nums[i-1]) continue
        let j=i+1
        let z=nums.length-1
        while(j<z){
            const sum = nums[i]+nums[j]+nums[z]
            if(sum===0){
                three.push([nums[i],nums[j],nums[z]])
                while(j<z&&nums[j]===nums[j+1]) j++
                while(j<z&&nums[z]===nums[z-1]) z--
                j++
                z--
            }else if(sum>0){
                z--
            }else{
                j++
            }
        }
    }
    return three
};

如果用暴力解法,三层循环时间复杂度是O(n³),这个算法的目的就是降低时间复杂度+去重

①为什么排序?使重复的值相邻,可以更好地跳过:数组.sort((a,b)=>a-b) 从小到大,升序

②去重:如果下一个值和上一个值相同,直接跳过,i++/j++/z++

注释版:

TypeScript 复制代码
function threeSum(nums: number[]): number[][] {
    //初始化声明二维数组,数组嵌套数组
    const three:number[][] = []
    //从小到大拍寻
    nums.sort((a,b)=>a-b)
    //i是第一个数的下标
    //i<length-2 判断条件,比如length=5,i<3,i的值是0,1,2 取不到3
    //给j、z腾两个位置
    for(let i=0;i<nums.length-2;i++){
        //如果重复,continue跳过下面步骤,直接i++
        if(i>0 && nums[i]===nums[i-1]) continue
        //定义一个左指针
        let j=i+1
        //定义一个右指针
        let z=nums.length-1
        //两个指针的不重合且一个在左一个在右
        while(j<z){
            const sum = nums[i]+nums[j]+nums[z]
            if(sum===0){
                //满足条件就push末尾添加
                three.push([nums[i],nums[j],nums[z]])
                //如果重复,再走一遍循环也是=0,j++移到最后一个相同的值的下标
                while(j<z&&nums[j]===nums[j+1]) j++
                //z同理
                while(j<z&&nums[z]===nums[z-1]) z--
                //以上步骤都是j和z都指向相同值得最后一个下标

                //指针需要继续移位查找更多的组合
                j++
                z--
            //如果和大了,证明需要更小的数,右指针需要往前移
            }else if(sum>0){
                z--
            //如果和小了,证明需要更大的数,左指针需要往前移
            }else{
                j++
            }
        }
    }
    return three
};

共勉,祝天下考生旗开得胜!

相关推荐
AI小老六9 小时前
SkillOpt 架构拆解:把 Skill 文本当参数,用执行轨迹训练 Agent
后端·算法·ai编程
胡萝卜术9 小时前
从“分数打架”到“排名投票”:为什么你的ChatBI必须用RRF?
算法·设计模式·面试
Asize10 小时前
初识DFS 与 BFS:递归、队列与图遍历
算法
罗西的思考1 天前
机器人 / 强化学习】HIL-SERL:人类在环驱动的具身智能进化框架
人工智能·算法·机器学习
kyriewen1 天前
别再对着 TypeScript 报错发呆了:我把 10 个最常见的红色波浪线翻译成了人话
前端·javascript·typescript
CSharp精选营1 天前
关系型 vs 非关系型:从原理到选型,一文搞定数据库核心分类
数据结构·nosql·关系型数据库·非关系型数据库·技术选型
美团技术团队1 天前
LongCat 开源 VitaBench 2.0:长期动态智能体基准新标杆
人工智能·算法
妙码生花1 天前
现代前端的极致性能 icon 加载方案(死磕成功版)
前端·vue.js·typescript
MonkeyKing1 天前
鸿蒙ArkTS深度剖析:ArkTS与TS/JS核心差异、静态强类型实战优势
typescript·harmonyos
To_OC2 天前
LC 207 课程表:刚学图论那会儿,我连这是拓扑排序都没看出来
javascript·算法·leetcode