力扣:两数之和与n数之和的(Map)与(排序+双指针)解法 【三刷终于明白HashMap求和的去重问题】

news2024/12/29 11:39:39

啃一本算法书啃了快一年了,用嘴想一想都该只剩渣了,脑子是怎么想的???

  • 真希望有一天“爷啃完了,爷不要你了,爷换一本啃”,,欸欸欸??罪过罪过!!!
  • 哈哈哈

文章目录

  • 一、 两数之和
    • 1-1
      • 1-1-1 排序+双指针【返回!!数组元素!!不是数组下标】
      • 1-1-2 HashMap [ 两数之和](https://leetcode.cn/problems/two-sum/)
  • 二、 三数之和【返回数组元素】
    • 2-1 [ 三数之和](https://leetcode.cn/problems/3sum/)
      • 2-1-1 排序+双指针
      • 2-1-2 思路分析
      • 2-1-3 去重逻辑的思考
        • 2-1-3-1 a的去重
        • 2-1-3-2 b与c的去重

一、 两数之和

1-1

  • 两数之和【返回数组下标】使用HashMap解决就很巴适,真简单!
  • 排序+双指针给n数之和铺路【返回数组元素】
    • 排序之后数组下标对应的元素发生改变!!!

1-1-1 排序+双指针【返回!!数组元素!!不是数组下标】

/**
 * @param {number[]} nums
 * @param {number} target
 * @return {number[]}
 */
var twoSum = function(nums, target) {
    // 当需要返回结果数组时
    // 排序+双指针
    nums.sort((a,b) => {return a-b;})
    let left = 0,right = nums.length-1;
    
    const res = [];

    // 左右指针
   while(left < right) {
       let leftValue = nums[left],rightValue = nums[right];
       let sum = leftValue + rightValue
       
       if(sum < target){
           while(left < right && nums[left] == leftValue) left++;  
       } 
       else if(sum > target) {
           while (left < right && nums[right] == rightValue)  right--;
       }
       else {
           res.push(nums[left],nums[right])
           while(left < right && nums[left] == leftValue ) left++;
           while(left < right && nums[right] == rightValue ) right--;
       }
   }

   return res;
};

1-1-2 HashMap 两数之和

  • 用 hashMap 存储遍历过的元素和对应的索引。
  • 每遍历一个元素,看看 hashMap 中是否存在满足要求的目标数字。
  • 所有事情在一次遍历中完成(空间换取时间)
/**
 * @param {number[]} nums
 * @param {number} target
 * @return {number[]}
 */
var twoSum = function(nums, target) {
   const map = new Map();
   for(let i in nums) {
    
    if(map.get(target - nums[i])) {
        return [i,map.get(target - nums[i])]
    }
    map.set(nums[i],i)
   } 
};

二、 三数之和【返回数组元素】

2-1 三数之和

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

2-1-1 排序+双指针

在这里插入图片描述

/**
 * @param {number[]} nums
 * @return {number[][]}
 */
var threeSum = function(nums) {
    return threeSumTarget(nums, 0)
};
const threeSumTarget = (nums, target) => {
    nums.sort((a,b)=>a-b)
    // const res = new Set();
    const res= [];
    for(let i = 0; i<nums.length; i++) {
        let temp = towSumTarget(nums, i+1, target - nums[i])
        if(temp.length) {
            for(let item of temp) {
                item.push(nums[i])
                res.push(item);
            }
        }
        // 跳过第一个数字重复的情况,否则会出现重复结果
        while(i < nums.length-1 && nums[i]==nums[i+1]) i++;
    }
    return res;
}
var towSumTarget = function(nums, start, target) {

    // 当需要返回结果数组时
    // 排序+双指针
    nums.sort((a,b) => {return a-b;})
    let left = start,right = nums.length-1;
    
    const res = [];

    // 左右指针
   while(left < right) {
       let leftValue = nums[left];
       let rightValue = nums[right];
       if(nums[left] + nums[right] < target) left++;
       else if(nums[left] + nums[right] > target) right--;
       else {
           res.push([nums[left],nums[right]])
           while(left < right && nums[left] == leftValue ) left++;
           while(left < right && nums[right] == rightValue ) right--;
       }
   }

   return res;
};




2-1-2 思路分析

  1. 确定 第一个数字nums[i]之后,剩下的两个数字就是和为target-nums[i]的两个数字

    • 三数之和------》 两数之和
  2. 请添加图片描述

2-1-3 去重逻辑的思考

说道去重,其实主要考虑三个数的去重。 a, b ,c, 对应的就是 nums[i],nums[left],nums[right]

2-1-3-1 a的去重

  1. 分析

    • a 如果重复了怎么办,a是nums里遍历的元素,那么应该直接跳过去。

      • 但这里有一个问题,是判断 nums[i] 与 nums[i + 1]是否相同,还是判断 nums[i] 与 nums[i-1] 是否相同。

      • 都是和 nums[i]进行比较,是比较它的前一个,还是比较他的后一个。

        if (nums[i] == nums[i + 1]) { // 去重操作
            continue;
        }
        
    • 那就我们就把 三元组中出现重复元素的情况直接pass掉了。 例如{-1, -1 ,2} 这组数据,当遍历到第一个-1 的时候,判断 下一个也是-1,那这组数据就pass了。

      • 我们要做的是 不能有重复的三元组,但三元组内的元素是可以重复的!
  2. 所以这里是有两个重复的维度。那么应该这么写:

if (i > 0 && nums[i] == nums[i - 1]) {
    continue;
}
  • 这么写就是当前使用 nums[i],我们判断前一位是不是一样的元素,在看 {-1, -1 ,2} 这组数据,当遍历到 第一个 -1 的时候,只要前一位没有-1,那么 {-1, -1 ,2} 这组数据一样可以收录到 结果集里。

2-1-3-2 b与c的去重

  1. 很多同学写本题的时候,去重的逻辑多加了 对right 和left 的去重:(代码中注释部分)
while (right > left) {
    if (nums[i] + nums[left] + nums[right] > 0) {
        right--;
        // 去重 right
        while (left < right && nums[right] == nums[right + 1]) right--;
    } else if (nums[i] + nums[left] + nums[right] < 0) {
        left++;
        // 去重 left
        while (left < right && nums[left] == nums[left - 1]) left++;
    } else {
    }
}
  1. 这种去重其实对提升程序运行效率是没有帮助的。

    • 拿right去重为例,即使不加这个去重逻辑,依然根据 while (right > left) 和 if (nums[i] + nums[left] + nums[right] > 0) 去完成right-- 的操作。
    • 多加了 while (left < right && nums[right] == nums[right + 1]) right–; 这一行代码,其实就是把 需要执行的逻辑提前执行了,但并没有减少 判断的逻辑。
    • 最直白的思考过程,就是right还是一个数一个数的减下去的,所以在哪里减的都是一样的。

    • 所以这种去重 是可以不加的。 仅仅是 把去重的逻辑提前了而已。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Python基础(一)

Python 的种类 CpythonPython的官方版本&#xff0c;使用C语言实现&#xff0c;使用最为广泛&#xff0c;CPython实现会将源文件&#xff08;py文件&#xff09;转换成字节码文件&#xff08;pyc文件&#xff09;&#xff0c;然后运行在Python虚拟机上。 JyhtonPython的Java实…

IDEA初始化git+代码提交

IDEA初始化git 当你在代码仓库如&#xff1a;github or gitee 上建立好了仓库&#xff0c;我们在idea中该如何初始化你的git又怎么样把你的代码push到代码仓库上呢&#xff1f; 第一步&#xff1a;初始化idea中的git 在idea中的setting中搜索git&#xff0c;将你的git.exe路…

JVM垃圾回收器-评估GC的性能指标

文章目录学习资料垃圾回收器概述评估GC的性能指标吞吐量&#xff08;throughput&#xff09;暂停时间&#xff08;pause time&#xff09;吞吐量VS暂停时间学习资料 【尚硅谷宋红康JVM全套教程&#xff08;详解java虚拟机&#xff09;】 【阿里巴巴Java开发手册】https://www.…

机器学习 加利福尼亚房价预测

学习目标&#xff1a; 提示&#xff1a;导入包 例如&#xff1a; import pandas as pd import numpy as npfrom sklearn.datasets import fetch_california_housing from sklearn.model_selection import KFold, train_test_split from sklearn.metrics import mean_squared…

【阶段二】Python数据分析Pandas工具使用02篇:数据读取:文本文件读取、电子表格读取与数据预处理:数据概览与清洗

本篇的思维导图: 数据读取:文本文件读取 对于csv后缀的文本文件,可以使用pandas模块中的read_csv函数进行读取。 所需要的数据文件如下百度云盘链接: 链接:https://pan.baidu.com/s/1Zj-uTt_wdRcmDt3aumZ2nA 提取码:z2e8 代码

CSRF攻防基础讲解

CSRF攻击 Cross-site request forgery跨站请求伪造 场景模拟 在用户登录某个网站后&#xff0c;看到某篇文章高兴之余&#xff0c;挥手打字&#xff0c;突然有人发来一个链接&#xff0c;登录者打开链接后什么都没有操作或者只是好奇的点击了某个按钮&#xff0c;在原登录网…

猿客栈后台管理系统日志记录

目录 一、用户权限设置 前端逻辑 后端逻辑 二、登录界面逻辑 1、账号密码登录实现 前端逻辑 在Cookie中存储token的方法 在Cookie中存储和获取的token方法 后端逻辑 生成token工具类 2、手机号登录 前台逻辑 后台逻辑 补充&#xff1a;实现点击发送验证码120s倒计…

ThinkPHP 之 SQLI审计分析(二)

说明 该文章来源于同事lu2ker转载至此处&#xff0c;更多文章可参考&#xff1a;https://github.com/lu2ker/ 文章目录说明0x00 测试代码做了什么&#xff1f;0x01 分析调用0x02 漏洞点的发现、构造、利用0x03 总结Time&#xff1a;9-3影响版本&#xff1a;ThinkPHP5.0.10 Pa…

详细介绍chrony服务器

chrony服务器 硬件时间&#xff1a;BIOS里面&#xff1b;关机后依然运行&#xff0c;主板电池为它供电&#xff1b;RTC时钟 系统时间&#xff1a;开机后&#xff0c;软件启动读取硬件时钟&#xff0c;之后独立运行 Chrony 的配置文件是/etc/chrony.conf chronyd服务器端 ch…

【目标检测】Mask rcnn代码实现Pytorch版,适用30系列显卡!(测试版)

目录&#xff1a;Mask rcnn代码实现Pytorch版一、环境二、mmdetection环境搭建三、测试四、结果展示为什么选择使用Pytorch版本&#xff1f;因为本人换电脑了&#xff0c;显卡升级为30系列&#xff0c;而30系列显卡的 CUDA 版本要求是 11.x。一、环境 cudatoolkit …

MYSQL之两阶段提交和组提交(数据一致性)

我们在MySQL 的日志中详细的介绍了undo log、redo log、binlog这三个日志和所用到的一些缓存知识&#xff0c;那么下面我们分析一下更新语句执行过程&#xff0c;它们是怎么变化的呢&#xff1f;下面我们直接给答案吧。假如我们修改一行主键索引&#xff08;id&#xff09;为1的…

电力系统激励型需求响应+自适应多群体优化算法(Python实现)

目录 ​编辑 0 前言 1 激励型DR和价格型DR 2 激励型DR模型 3 Python代码实现 4 自适应多群体优化算法&#xff08;AMPO&#xff09; 5 Python代码实现 0 前言 风、光等清洁能源因具有环保、资源丰富等优点而受到电力行业的重视,电力行业开始大力发展清洁能源发电。同时…

MySQL详解,库和表的基础操作

目录 前言 一、预备知识 1、服务器&#xff0c;数据库&#xff0c;表关系 2、SQL分类 3、连接服务器 二、库的操作 1、创建数据库 2、查看字符集和校验规则 2.1 查看系统默认字符集以及校验规则 2.2 查看数据库支持的字符集和字符集校验规则 3、操纵数据库 3.1查看…

ros tf坐标

参考&#xff1a; 讲解&#xff1a;https://www.bilibili.com/video/BV1zt411G7Vn/?p18&vd_source3a1ad336af3eaae4fcced56c75d309d1ROS程序&#xff1a;https://gitee.com/guyuehome/ros_21_tutorials/tree/master/learning_tfROS2程序&#xff1a;https://gitee.com/gu…

公司企业兔年祝福元旦祝福贺卡邀请函模板!

能群发的贺卡邀请函如何制作&#xff1f;想制作一个专属的祝福贺卡邀请函有什么方法&#xff1f;下面跟着小编的乔拓云工具教程&#xff01;教你如何使用这个工具在线就能轻松搞定设计需求&#xff0c;不仅有海量模板供你使用&#xff0c;还能一键生成链接轻松转发&#xff01;…

利用vue-cli创建vue3工程

需注意&#xff1a;想创建vue3工程&#xff0c;对vue-cli版本有要求&#xff0c;必须确保vue-cli在4.5.0以上 目录 1、查看vue-cli版本 2、创建工程 3、启动 1、查看vue-cli版本 vue --version&#xff08;小写v&#xff09; vue --version或者 vue -V&#xff08;大写v&…

深入分析Java中finalize方法的作用和底层原理

finalize方法是什么 finalize方法是Object的protected方法&#xff0c;Object的子类们可以覆盖该方法以实现资源清理工作&#xff0c;GC在首次回收对象之前调用该方法。 finalize方法与C的析构函数的区别 finalize方法与C中的析构函数不是对应的&#xff0c;C中的析构函数调…

(小程序)后台交互-首页

目录 一、小程序首页动态数据加载 1.数据库准备 2.后台准备 ① pom.xml ② 配置数据源 ③ 整合mybatis ④ 代码生成 ⑤ mybatis-generator 二、准备前端的首页的数据 1、Promise 2.封装request 3.会议展示 三、通过wxs将首页动态数据优化 一、小程序首页动态数据加…

Git命令版(powernode)

Git命令版&#xff08;powernode&#xff09; 目录Git命令版&#xff08;powernode&#xff09;1.添加文件相关命令案例实操小结&#xff1a;2.工作区和暂存区2.1 名词解释。2.1.1 工作区&#xff08;Working Directory&#xff09;2.1.2 版本库&#xff08;Repository&#xf…

数据结构之双向链表

双向链表与单向链表较为类似&#xff0c;单向链表有一个指针域&#xff0c;用来指向后继结点&#xff0c;而双向链表有两个指针域&#xff0c;分别用来指向前驱结点和后继结点。玩双向链表时一定要从单向链表的思维中跳出来&#xff0c;否则在操作双向链表时就会出现各种问题。…