【优选算法之双指针】No.2--- 经典双指针算法(下)

文章目录

  • 前言
  • 一、双指针示例:
    • 1.1 ⽔果成篮
    • 1.2 和为s的两个数字
    • 1.3 三数之和
    • 1.4 四数之和
  • 二、双指针总结:


前言

在这里插入图片描述

👧个人主页:@小沈YO.
😚小编介绍:欢迎来到我的乱七八糟小星球🌝
📋专栏:优选算法
🔑本章内容:双指针
记得 评论📝 +点赞👍 +收藏😽 +关注💞哦~


提示:以下是本篇文章正文内容,下面案例可供参考

一、双指针示例:

1.1 ⽔果成篮

  1. 题⽬链接:611. 有效三⻆形的个数
  2. 题⽬描述:
    在这里插入图片描述
  3. 解法⼀(暴⼒求解)(会超时):
    算法思路:三层 for 循环枚举出所有的三元组,并且判断是否能构成三⻆形。
    虽然说是暴⼒求解,但是还是想优化⼀下:
    判断三⻆形的优化:
  • 如果能构成三⻆形,需要满⾜任意两边之和要⼤于第三边。但是实际上只需让较⼩的两条边之和⼤于第三边即可。
  • 因此我们可以先将原数组排序,然后从⼩到⼤枚举三元组,⼀⽅⾯省去枚举的数量,另⼀⽅⾯⽅便判断是否能构成三⻆形
  1. 解法⼆(排序 + 双指针):
    算法思路:先将数组排序。
    根据「解法⼀」中的优化思想,我们可以固定⼀个「最⻓边」,然后在⽐这条边⼩的有序数组中找出⼀个⼆元组,使这个⼆元组之和⼤于这个最⻓边。由于数组是有序的,我们可以利⽤「对撞指针」来优化。
    设最⻓边枚举到 i 位置,区间 [left, right] 是 i 位置左边的区间(也就是⽐它⼩的区间):
    ◦ 如果 nums[left] + nums[right] > nums[i] :
    ▪ 说明 [left, right - 1] 区间上的所有元素均可以与 nums[right] 构成⽐ nums[i] ⼤的⼆元组
    ▪ 满⾜条件的有 right - left 种
    ▪ 此时 right 位置的元素的所有情况相当于全部考虑完毕, right-- ,进⼊下⼀轮判断
    ◦ 如果 nums[left] + nums[right] <= nums[i] :
    ▪ 说明 left 位置的元素是不可能与 [left + 1, right] 位置上的元素构成满⾜条件的⼆元组
    ▪ left 位置的元素可以舍去, left++ 进⼊下轮循环
  2. C++代码(数组模拟哈希/容器)
class Solution {
public:int triangleNumber(vector<int>& nums) {int cnt=0;sort(nums.begin(),nums.end());if(nums.size()<3)return 0;for(int i=nums.size()-1;i>=2;i--){int left=0,right=i-1;while(left<right){if(nums[left]+nums[right]>nums[i]){cnt+=right-left;right--;}else{left++;}}}return cnt;}
};

1.2 和为s的两个数字

  1. 题⽬链接: 剑指 Offer 57. 和为s的两个数字

  2. 题⽬描述:
    在这里插入图片描述

  3. 解法⼀(暴⼒解法,会超时):
    算法思路:两层 for 循环列出所有两个数字的组合,判断是否等于⽬标值。
    算法流程:
    两层 for 循环:
    ◦ 外层 for 循环依次枚举第⼀个数 a ;
    ◦ 内层 for 循环依次枚举第⼆个数 b ,让它与 a 匹配;
    ps :这⾥有个魔⻤细节:我们挑选第⼆个数的时候,可以不从第⼀个数开始选,因为 a 前⾯的数我们都已经在之前考虑过了;因此,我们可以从 a 往后的数开始列举。
    ◦ 然后将挑选的两个数相加,判断是否符合⽬标值

  4. 解法⼆(双指针 - 对撞指针):
    算法思路:
    注意到本题是升序的数组,因此可以⽤「对撞指针」优化时间复杂度。算法流程(附带算法分析,为什么可以使⽤对撞指针):
    a. 初始化 left , right 分别指向数组的左右两端(这⾥不是我们理解的指针,⽽是数组的下标)
    b. 当 left < right 的时候,⼀直循环
    i. 当 array[left] + array[right] == sum时,说明找到结果,记录结果,并且返回;
    ii. 当 array[left] + array[right] < sum时:
    • 对于 array[left] ⽽⾔,此时 array[right] 相当于是 array[left] 能碰到的最⼤值(别忘了,这⾥是升序数组哈~)。如果此时不符合要求,说明在这个数组⾥⾯,没有别的数符合 nums[left] 的要求了(最⼤的数都满⾜不了你,你已经没救了)。因此,我们可以⼤胆舍去这个数,让 left++ ,去⽐较下⼀组数据;
    • 那对于 array[right] ⽽⾔,由于此时两数之和是⼩于⽬标值的, array[right] 还可以选择⽐ array[left] ⼤的值继续努⼒达到⽬标值,因此 right 指针我们按兵不动;
    iii. 当 array[left] + array[right] > sum时,同理我们可以舍去 array[right] (最⼩的数都满⾜不了你,你也没救了)。让 right-- ,继续⽐较下⼀组数据,⽽ left 指针不变(因为他还是可以去匹配⽐ array[right] 更⼩的数的)

  5. C++代码

class Solution {
public:vector<int> FindNumbersWithSum(vector<int> array,int sum) {sort(array.begin(),array.end());vector<int> v;int left=0,right=array.size()-1;while(left<right){if(array[left]+array[right]<sum)left++;else if(array[left]+array[right]>sum)right--;else{v.push_back(array[left]);v.push_back(array[right]);break;}}return v;}
};

1.3 三数之和

  1. 题⽬链接:15. 三数之和

  2. 题⽬描述:
    在这里插入图片描述

  3. 解法(排序+双指针):
    算法思路:本题与两数之和类似,是⾮常经典的⾯试题
    与两数之和稍微不同的是,题⽬中要求找到所有「不重复」的三元组。那我们可以利⽤在两数之和那⾥⽤的双指针思想,来对我们的暴⼒枚举做优化:
    i. 先排序;
    ii. 然后固定⼀个数 a :
    iii. 在这个数后⾯的区间内,使⽤「双指针算法」快速找到两个数之和等于 -a 即可。
    但是要注意的是,这道题⾥⾯需要有「去重」操作~
    i. 找到⼀个结果之后, left 和 right 指针要「跳过重复」的元素;
    ii. 当使⽤完⼀次双指针算法之后,固定的 a 也要「跳过重复」的元素

  4. C++代码

class Solution {
public:vector<vector<int>> threeSum(vector<int>& nums) {vector<vector<int>> vv;sort(nums.begin(),nums.end());set<vector<int>> sv;for(int i=nums.size()-1;i>=2;){int left=0,right=i-1;while(left<right){if(nums[left]+nums[right]<(-nums[i])){left++;}else if(nums[left]+nums[right]>(-nums[i])){right--;}else{vv.push_back({nums[left],nums[right],nums[i]});left++;right--;while(left<right&&nums[left]==nums[left-1])left++;while(left<right&&nums[right]==nums[right+1])right--;}}i--;while(i>=2&&nums[i]==nums[i+1])i--;}return vv;}
};

1.4 四数之和

  1. 题⽬链接:18. 四数之和
  2. 题⽬描述:
    在这里插入图片描述
  3. 解法(排序 + 双指针)
    算法思路:
    a. 依次固定⼀个数 a ;
    b. 在这个数 a 的后⾯区间上,利⽤「三数之和」找到三个数,使这三个数的和等于 target- a 即可。
  4. C++代码
class Solution {
public:vector<vector<int>> fourSum(vector<int>& nums, int target) {vector<vector<int>> vv;if(nums.size()<4)return vv;sort(nums.begin(),nums.end());for(int dest=nums.size()-1,cur=dest-1;dest>=3;){while(cur>=2){int left=0,right=cur-1;while(left<right){long long sum=nums[left]+nums[right];long long tmp=(long long)target-nums[dest]-nums[cur];if(sum<tmp){left++;}else if(sum>tmp){right--;}else{vv.push_back({nums[left],nums[right],nums[cur],nums[dest]});left++;right--;while(left<right&&nums[right]==nums[right+1])right--;while(left<right&&nums[left]==nums[left-1])left++;}}cur--;while(cur>=2&&nums[cur]==nums[cur+1])cur--;}dest--;while(dest>=3&&nums[dest]==nums[dest+1])dest--;cur=dest-1;}return vv;}
};

二、双指针总结:

  • 1.移动零:快排的思想:数组划分区间 - 数组分两块
  • 2.复写零:从后往前(涉及到覆盖的)
  • 3.快乐数:快慢指针(设计循环)
  • 4.盛水最多的容器:对撞指针
  • 5.有效三角的个数:固定一边+对撞指针
  • 6.和为S的两个数字:排序+对撞指针
  • 7.三数之和:排序+对撞指针+固定
  • 8.四数之和:排序+对撞指针(三指针)+固定

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

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

相关文章

安装黑群晖系统,并使用NAS公网助手访问教程(好文)

由于正版群晖系统的价格不菲&#xff0c;对于预算有限的用户来说&#xff0c;安装黑群晖系统成为了一个不错的选择&#xff08;如果您预算充足&#xff0c;建议选择白群晖&#xff09;。如您对宅系科技比较感兴趣&#xff0c;欢迎查看本文&#xff0c;将详细介绍如何安装黑群晖…

allWebPlugin中间件自定义alert、confirm及prompt使用

allWebPlugin简介 allWebPlugin中间件是一款为用户提供安全、可靠、便捷的浏览器插件服务的中间件产品&#xff0c;致力于将浏览器插件重新应用到所有浏览器。它将现有ActiveX控件直接嵌入浏览器&#xff0c;实现插件加载、界面显示、接口调用、事件回调等。支持Chrome、Firefo…

跨游戏引擎的H5渲染解决方案(腾讯)

本文是腾讯的一篇H5 跨引擎解决方案的精炼。 介绍 本文通过实现基于精简版的HTML5&#xff08;HyperText Mark Language 5&#xff09;来屏蔽不同引擎&#xff0c;平台底层的差异。 好处&#xff1a; 采用H5的开发方式&#xff0c;可以将开发和运营分离&#xff0c;运营部门自…

代码随想录Day 51|题目:99.岛屿数量、100.岛屿的最大面积

提示&#xff1a;DDU&#xff0c;供自己复习使用。欢迎大家前来讨论~ 文章目录 题目一&#xff1a;99. 岛屿数量思路深度优先搜索DFS广度优先搜索BFS 题目二&#xff1a;100. 岛屿的最大面积DFSBFS 总结 题目一&#xff1a;99. 岛屿数量 99. 岛屿数量 (kamacoder.com) 思路 …

Tomcat服务器—Windows下载配置详细教程

一、关于 1.1 简介 Tomcat是一个开源的Java Servlet容器和Web服务器&#xff0c;由Apache软件基金会维护。它实现了Java Servlet和JavaServer Pages (JSP) 规范&#xff0c;用于运行Java Web应用程序。Tomcat支持多种Java EE功能&#xff0c;并提供了高效的性能和可扩展性&am…

华为OD机试 - 分解正整数 - 数学推导(Python/JS/C/C++ 2024 D卷 100分)

华为OD机试 2024E卷题库疯狂收录中&#xff0c;刷题点这里 专栏导读 本专栏收录于《华为OD机试真题&#xff08;Python/JS/C/C&#xff09;》。 刷的越多&#xff0c;抽中的概率越大&#xff0c;私信哪吒&#xff0c;备注华为OD&#xff0c;加入华为OD刷题交流群&#xff0c;…

Redisson实现分布式锁(看门狗机制)

目录 可重入锁&#xff1a; 锁重试和看门狗机制&#xff1a; 主从一致性&#xff1a; 首先引入依赖&#xff0c;配置好信息 3.使用Redisson的分布式锁 可重入锁&#xff1a; 可重入锁实现是通过redsi中的hash实现的&#xff0c;key依旧是业务名称加id&#xff0c;然后第一个…

如何成立一家自己的等级保护测评机构?需要哪些条件?有哪些要求?

给大家的福利&#xff0c;点击下方蓝色字 即可免费领取↓↓↓ &#x1f91f; 基于入门网络安全/黑客打造的&#xff1a;&#x1f449;黑客&网络安全入门&进阶学习资源包 前言 各省、自治区、直辖市公安厅、局网络安全保卫总队&#xff0c;新疆生产建设兵团公安局网络安…

【高分系列卫星简介——高分一号(GF-1)】

高分一号卫星&#xff08;GF-1&#xff09; 高分一号&#xff08;GF-1&#xff09;是中国高分辨率对地观测系统&#xff08;简称“高分专项”&#xff09;的第一颗卫星&#xff0c;具有里程碑式的意义。以下是对高分一号卫星的详细介绍&#xff1a; 一、基本信息 发射时间&…

2024华为杯研究生数学建模竞赛(研赛)选题建议+初步分析

提示&#xff1a;C君认为的难度&#xff1a;DE<C<F&#xff0c;开放度&#xff1a;CDE>F。 华为专项的题目&#xff08;A、B题&#xff09;暂不进行选题分析&#xff0c;不太建议大多数同学选择&#xff0c;对自己专业技能有很大自信的可以选择华为专项的题目。后续会…

MyBatis-config.xml核心配置

MyBatis-config.xml 包含了会深深影响MyBatis行为的设置和属性信息&#xff0c;配置文档的顶层结构如下 environments&#xff08;环境配置&#xff09; environments用于配置数据库的URL信息&#xff0c;MyBatis-config可以动态配置多个数据源&#xff0c;用于连生产、预发、…

用 HTML + JavaScript DIY 一个渐进式延迟法定退休年龄测算器

为减轻社会和个人因退休年龄变化带来的冲击&#xff0c;近日&#xff0c;全国人民代表大会常务委员会正式发布了关于实施渐进式延迟法定退休年龄的重要决定。 根据该决定&#xff0c;我国将同步启动对男、女职工法定退休年龄的延迟计划。这一调整将采取渐进式的方式进行&#…

RabbitMQ 高级特性——发送方确认

文章目录 前言发送方确认confirm 确认模式return 退回模式 常见面试题 前言 前面我们学习了 RabbitMQ 中交换机、队列和消息的持久化&#xff0c;这样能够保证存储在 RabbitMQ Broker 中的交换机和队列中的消息实现持久化&#xff0c;就算 RabbitMQ 服务发生了重启或者是宕机&…

Nginx实用篇:实现负载均衡、限流与动静分离

Nginx实用篇&#xff1a;实现负载均衡、限流与动静分离 | 原创作者/编辑&#xff1a;凯哥Java | 分类&#xff1a;Nginx学习系列教程 Nginx 作为一款高性能的 HTTP 服务器及反向代理解决方案&#xff0c;在互联网架构中扮演着至关重要的角色。它…

Acwing DFS

DFS&#xff1a;深度优先搜索 DFS与BFS的对比 DFS使用栈来实现&#xff0c;BFS使用队列来实现 DFS所需要的空间是 O ( h ) O(h) O(h),而BFS需要的空间是 O ( 2 h ) O(2^h) O(2h),其中h是树的高度&#xff1b; DFS不具有最短路的特性&#xff0c;BFS有最短路的特性 DFS回溯…

102.SAPUI5 sap.ndc.BarcodeScannerButton调用摄像头时,localhost访问正常,使用IP访问失败

目录 原因 解决办法 1.修改谷歌浏览器的setting 2.在tomcat中配置https访问 参考 使用SAPUI5的sap.ndc.BarcodeScannerButton调用摄像头时&#xff0c;localhost访问正常&#xff0c;使用IP访问时&#xff0c;一直打不开摄像头&#xff0c;提示getUserMedia()问题。 原因…

2024 “华为杯” 中国研究生数学建模竞赛(D题)深度剖析|大数据驱动的地理综合问题|数学建模完整代码+建模过程全解全析

当大家面临着复杂的数学建模问题时&#xff0c;你是否曾经感到茫然无措&#xff1f;作为2022年美国大学生数学建模比赛的O奖得主&#xff0c;我为大家提供了一套优秀的解题思路&#xff0c;让你轻松应对各种难题&#xff01; CS团队倾注了大量时间和心血&#xff0c;深入挖掘解…

elasticsearch同步mysql方案

文章目录 1、1. 使用数据库触发器2. 使用定时任务3. 监听MySQL二进制日志&#xff08;binlog&#xff09;4. 使用数据管道5. 使用第三方工具或服务6. 编写自定义脚本注意事项 2、1. 使用Logstash步骤&#xff1a;示例配置&#xff1a; 2. 使用Debezium步骤&#xff1a; 3. 自定…

828华为云征文 | 解锁企业级邮件服务,在华为云fFlexus x实例上部署Mailcow开源方案

前言 华为云Flexus X实例携手Mailcow开源邮件方案&#xff0c;为企业打造了一个既高效又安全的邮件服务解决方案。Flexus X实例的柔性算力与高性能&#xff0c;是这一方案的坚实基石。它提供CPU内存的灵活定义&#xff0c;以经济型价格实现旗舰级性能&#xff0c;确保邮件服务的…

云计算课程作业1

作业1 Xmanager连接 rhel连接 作业2 首先确认你的虚拟机设置的是NAT 1-3 然后打开这篇blog&#xff0c;并完成第一步和第二步 因为我们是NAT&#xff0c;所以不需要连接网桥&#xff0c;即跳过第三步&#xff0c;但是这里ping一下测试网络连接 2- 如果到这里你发现提示yum…