算法魅力-双指针的实战

目录

1.双指针的介绍

 1. 左右指针(对撞指针)

 2. 快慢指针

2.题目练习讲解

 2.1 移动零

 算法思路

代码展示

画图效果效果

2.2 复写零

算法思路

 代码展示

2.3 快乐数

算法思路

代码展示

2.4 盛最多水的容器

算法思路

代码展示

结束语


1.双指针的介绍

双指针算法是一种常用的算法技巧,通常用于解决数组或链表相关的题目。双指针算法的核心思想是使用两个指针在数组或链表上移动,这里的指针并不是只是指指针,我们可以用数组下标来代替,以达到解决问题的目的。根据具体问题,双指针可以分为以下几种类型:

 1. 左右指针(对撞指针)

左右指针通常用于处理数组中的问题,其中一个指针从数组的开始位置向后移动,另一个指针从数组的结束位置向前移动。

指针从两端向中间移动。一个指针从最左端开始,另一个从最右端开始,然后逐渐往中间逼
对撞指针的 终止条件一般是两个指针相遇或者错开(也可能在循环内部找到结果直接跳出循环),也就是:
left == right (两个指针指向同一个位置)
left > right (两个指针错开)

典型问题:
二分查找:在有序数组中查找一个特定的元素。
两数之和:在数组中找到两个数,使它们的和等于一个给定的数。

 2. 快慢指针

快慢指针主要用于处理链表中的问题,两个指针从同一位置出发,一个指针(快指针)每次移动两个节点,另一个指针(慢指针)每次移动一个节点。
 典型问题:
判断链表中是否有环:Floyd 判圈算法(龟兔赛跑算法)。
找到链表的中间节点:快指针到达终点时,慢指针正好在中间。


双指针算法的关键在于指针的移动策略和终点的判断条件,根据具体问题,双指针的移动策略和判断条件可能会有所不同。

2.题目练习讲解

 2.1 移动零

283. 移动零 - 力扣(LeetCode)

给定一个数组 nums,编写一个函数将所有 0 移动到数组的末尾,同时保持非零元素的相对顺序。

请注意 ,必须在不复制数组的情况下原地对数组进行操作。

示例 1:

输入: nums = [0,1,0,3,12]输出: [1,3,12,0,0]

示例 2:

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

 算法思路

在本题中,我们可以用一个 cur 指针来扫描整个数组,另一个 dest 指针用来记录非零数序列
的最后一个位置。根据 cur 在扫描的过程中,遇到的不同情况,分类处理,实现数组的划分。
在 cur 遍历期间,使 [0, dest] 的元素全部都是非零元素, [dest + 1, cur - 1] 的元素全是零。
[cur,n-1]是待处理的元素。
首先,我们,我们让dest指向-1的位置,cur指向数组的首元素,通过循环控制cur的移动,无论cur指向谁,一次循环过后都要++,而当cur指向的数据是非0元素时,我们就让dest++,让加之后的dest与cur进行元素交换。
详细就是:
 cur 依次往后遍历每个元素,遍历到的元素会有下面两种情况:
i. 遇到的元素是 0 , cur 直接 ++ 。因为我们的目标是让 [dest + 1, cur - 1] 内
的元素全都是零,因此当 cur 遇到 0 的时候,直接 ++ ,就可以让 0 在 cur - 1
的位置上,从而在 [dest + 1, cur - 1] 内;
ii. 遇到的元素不是 0 , dest++ ,并且交换 cur 位置和 dest 位置的元素,之后让
cur++ ,扫描下⼀个元素。
• 因为 dest 指向的位置是非零元素区间的最后一个位置,如果扫描到一个新的非零元
素,那么它的位置应该在 dest + 1 的位置上,因此 dest 先自增 1 ;
• dest++ 之后,指向的元素就是 0 元素(因为非零元素区间末尾的后一个元素就是
0 ),因此可以交换到 cur 所处的位置上,实现 [0, dest] 的元素全部都是非零
元素, [dest + 1, cur - 1] 的元素全是零。

代码展示

class Solution
{
public:void moveZeroes(vector<int>& nums) {for(int cur = 0, dest = -1; cur < nums.size(); cur++)if(nums[cur]) // 处理⾮零元素swap(nums[++dest], nums[cur]);}
};

 当然,我们也可以让dest指向首元素,后续算法逻辑类似,只是变成了先交换在++。

class Solution
{
public:void moveZeroes(vector<int>& nums) {for(int cur = 0, dest = 0; cur < nums.size(); cur++)if(nums[cur]) // 处理⾮零元素swap(nums[dest++], nums[cur]);}
};

画图效果效果

de25a4d5a18f4260be424b6d85a6720f.png

2.2 复写零

1089. 复写零 - 力扣(LeetCode)

4d9f4b3c7c994c0fb91ec21a377ffce7.png

算法思路

同样这道题我们用双指针的算法,我们首先普遍会想到定义两个指针,从前向后开始复写,从首元素开始移动,在移动过程中由于0会写两次,我们会发现后一个数据会被覆盖就会达不到效果。

故当我们换成从后向前复写时,可以避免这种情况,但是我们需要找到复写的最后一个元素。

我们还是定义两个指针:

初始化两个指针 cur = 0 , dest = -1 ;
b. 找到最后一个复写的数:
i. 当 cur < n 的时候,一直执行下面循环:
• 判断 cur 位置的元素:
◦ 如果是 0 的话, dest 往后移动两位;
◦ 否则, dest 往后移动一位。
• 判断 dest 时候已经到结束位置,如果结束就终止循环;
• 如果没有结束, cur++ ,继续判断。
从 cur 位置开始往前遍历原数组,依次还原出复写后的结果数组:
i. 判断 cur 位置的值:
1. 如果是 0 : dest 以及 dest - 1 位置修改成 0 , dest -= 2 ;
2. 如果⾮零: dest 位置修改成 0 , dest -= 1 ;
ii. cur-- ,复写下一个位置。

while(cur>=0){if(arr[cur])arr[dest--]=arr[cur--];else{arr[dest--]=0;arr[dest--]=0;cur--;}
}

我们发现有些情况会数组越界,超出一位,当我们往前遍历复写的时候就会出现问题,因为数组外赋值了。

a03b657940be491e93dce4e78332a68a.png

所以我们要处理数组越界的情况

 如果越界,n - 1 位置的值修改成 0 ;  cur 向移动一步; dest 向前移动两步。

 代码展示

class Solution {
public:void duplicateZeros(vector<int>& arr) {int cur=0,dest=-1;int n=arr.size();while(cur<n){if(arr[cur])dest++;elsedest+=2;if(dest>=n-1)break;cur++;}if(dest==n){arr[n-1]=0;dest-=2;cur--;}while(cur>=0){if(arr[cur])arr[dest--]=arr[cur--];else{arr[dest--]=0;arr[dest--]=0;cur--;}}}
};

图片效果展示

6c2d65d18225463da479b00a3df4dc38.png

ad44edc2678a4272aa9835ee8ad52d0a.png

2.3 快乐数

202. 快乐数 - 力扣(LeetCode)

b5fccecf25aa4853b73494568c90c736.png

算法思路

首先可以将题目解答猜成两部分,第一部分是求取每个位数上的平方和,第二步就是与1进行比较。

通过快乐数的定义我们发现其实当它结果为1时,也会陷入一个1的循环。所以不管那种过程,累加次都会陷入循环,最终都会走到一个环中进行循环。

情况一:一直在 1 中死循环,即 1 -> 1 -> 1 -> 1......
情况二:在历史的数据中死循环,但始终变不到 1
3e8baa0eb30945f59793863ce51e0867.png

简单证明一下:

经过一次变化之后的最大值 9^2 * 10 = 810 ( 2^31-1=2147483647 。选一个更大的最
大 9999999999 ),也就是变化的区间在 [1, 810] 之间;  根据「鸽巢原理」,一个数变化 811 次之后,必然会形成一个循环;  因此,变化的过程最终会走到一个圈里面,因此可以用「快慢指针」来解决
将「对于一个正整数,每一次将该数替换为它每个位置上的数字的平方和」这一个
操作记为 x 操作
根据上述分析,我们可以知道,当重复执行 x 的时候,数据会陷入到一个「循环」之中。
而「快慢指针」有一个特性,就是在一个圆圈中,快指针总是会追上慢指针的,也就是说他们总会
相遇在一个位置上。如果相遇位置的值是 1 ,那么这个数一定是快乐数;如果相遇位置不是 1的话,那么就不是快乐数。

代码展示

class Solution {
public:int ret(int n){int sum=0;while(n){int a=n%10;sum+=a*a;n=n/10;}return sum;}bool isHappy(int n) {int slow=n;int fast=ret(n);while(slow!=fast){slow=ret(slow);fast=ret(ret(fast));}return slow==1;}
};

2.4 盛最多水的容器

11. 盛最多水的容器 - 力扣(LeetCode)

02f90008c9014914854eaa8f8660952f.png

47f59a7edd064a498fe284e80a40b605.png

60a0a16e37cd4ef290fc1e48bdf46bd8.png

这道题我们首先会想到枚举,通过暴力的解法将每一种的体积算出来,选出最大的。

lass Solution {
public:int maxArea(vector<int>& height) {int n=height.size();int ret=0;for(int i=0;i<n;i++){for(int j=i+1;j<n;j++){ret = max(ret, min(height[i], height[j]) * (j - i));}}return ret;}
};

 但是这种解法会超时。所以需要换一种思路。我们可以采用对撞指针的思路。

算法思路

设两个指针 left , right 分别指向容器的左右两个端点,此时容器的容积 :
v = (right - left) * min( height[right], height[left])
容器的左边界为 height[left] ,右边界为 height[right] 。
我们假设「左边边界」小于「右边边界」。
如果此时我们固定一个边界,改变另一个边界,水的容积会有如下变化形式:
容器的宽度一定变小。
由于左边界较小,决定了水的高度。 如果改变左边界,新的水面高度度不确定,但是一定不会超过右边的柱子高度,因此容器的容积可能会增大。
如果改变右边界,无论右边界移动到哪里, 新的水面的高度一定不会超过左边界,也就是不会超过现在的水面高度,但是由于容器的宽度减小,因此容器的容器一定会变小的。
由此可见,左边界和其余边界的组合情况都可以舍去。所以我们可以 left++ 跳过这个边界,继续去判断下一个左右边界。

代码展示

class Solution {
public:int maxArea(vector<int>& height) {int left=0,right=height.size()-1;int max=0;for(int i=0;i<height.size();i++){if(max<((right-left)*min(height[left],height[right])))max=(right-left)*min(height[left],height[right]);if(height[left]<=height[right])left++;elseright--;}return max;}
};

结束语

相信通过本篇博客大家对双指针算法有了一定的了解,下篇博客我们将继续讲解三道例题加深对双指针的印象。

最后感谢各位友友的支持,有不同的解法思路欢迎大家在评论区讨论!!!

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

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

相关文章

大模型带来新安全机遇

当前网络空间安全面临攻击隐蔽难发现、数据泄露风险高和违法信息审核难等挑战。大模型展现出强大的信息理解、知识抽取、意图和任务编排等能力&#xff0c;为网络空间安全瓶颈问题提供了新的解决思路和方法。与此同时&#xff0c;大模型发展也催生了恶意软件自动生成、深度伪造…

架构师之路-学渣到学霸历程-22

NFS文件共享服务器 今天开始了云计算-SRE架构师的第二个阶段&#xff0c; 第二阶段就是服务阶段了&#xff1b;第一个分享的就是NFS服务&#xff1b; 文件共享服务&#xff1b; 早上就了解一下NFS原理&#xff1b; 1、NFS文件共享服务器 NFS&#xff1a;就是network file sy…

【YOLO学习】YOLOv5详解

文章目录 1. 网络结构2. 结构整体描述2.1 输入端2.2 Backbone2.3 Neck2.4 Head 3. 模块细节3.1 Focus模块3.2 SPPF3.3 Bounding Box损失函数 4. 训练策略 1. 网络结构 1. 目标检测的模型框架大体都是以下图示这样的结构&#xff1a; 2. 关于 YOLOv5 的网络结构其实网上相关的讲…

数据结构 - 队列

队列也是一种操作受限的线性数据结构&#xff0c;与栈很相似。 01定义 栈的操作受限表现为只允许在队列的一端进行元素插入操作&#xff0c;在队列的另一端只允许删除操作。这一特性可以总结为先进先出&#xff08;First In First Out&#xff0c;简称FIFO&#xff09;。这意味…

R语言机器学习算法实战系列(八)逻辑回归算法 (logistic regression)

禁止商业或二改转载,仅供自学使用,侵权必究,如需截取部分内容请后台联系作者! 文章目录 介绍教程下载数据加载R包导入数据数据预处理数据描述数据切割构建模型预测测试数据评估模型模型准确性混淆矩阵模型评估指标ROC CurvePRC Curve特征的重要性保存模型总结系统信息介绍 …

MySQL数据库和表的基本操作

文章目录 一、数据库的基础知识 背景知识数据库的基本操作二、数据类型 字符串类型数值类型日期类型三、表的基本操作 创建表查看表结构查看所有表删除表 一、数据库的基础知识 背景知识 MySQL是一个客户端服务器结构的程序 主动发送数据的这一方&#xff0c;客户端(client…

“智改数转”转了什么?

万界星空科技专门针对数字化改造申报的MES系统具有显著的技术优势和实施效果&#xff0c;能够为制造型企业提供全方位、高效、可靠的数字化转型支持。项目合作可以私信或者百度上海万界星空科技官网。 “智改数转”是一个综合性的过程&#xff0c;涉及企业多个方面的转型和升…

【python实战】利用代理ip爬取Alibaba海外版数据

引言 在跨境电商的业务场景中&#xff0c;数据采集是分析市场、了解竞争对手以及优化经营策略的重要环节。然而&#xff0c;随着越来越多企业依赖数据驱动决策&#xff0c;许多跨境电商平台为了保护自身数据&#xff0c;采取了更严格的防护措施。这些平台通过屏蔽大陆IP地址或部…

【Spring声明式事务失效的12种场景测试】

文章目录 一.Spring声明式事务是什么&#xff1f;二.Spring事务失效的12种场景1.访问权限问题 小结 一.Spring声明式事务是什么&#xff1f; Spring声明式事务是一种通过配置的方式管理事务的方法&#xff0c;它通过注解或XML配置来声明哪些方法需要事务管理&#xff0c;从而将…

JRT怎么从IRIS切换到PostGreSql库

1.执行M导出得到建库脚本文件 2.下载生成的脚本到本地D盘 3.修改驱动为PostGreSql 4.修改连接串 5.到PostGreSql里面创建一个jrtlis的数据库&#xff0c;模式为jrt 6.启动网站点击导入脚本按钮 导入完成了就可以正常使用PostGreSql库了

OpenCV高级图形用户界面(14)交互式地选择一个或多个感兴趣区域函数selectROIs()的使用

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 允许用户在给定的图像上选择多个 ROI。 该函数创建一个窗口&#xff0c;并允许用户使用鼠标来选择多个 ROI。控制方式&#xff1a;使用空格键或…

Google FabricDiffusion:开启3D虚拟试穿新篇章

随着数字化转型的步伐不断加快,时尚界也在探索如何利用最新技术为消费者带来更加沉浸式的购物体验。在这一背景下,Google 推出了一项名为 FabricDiffusion 的新技术,这项技术能够将2D服装图像中的高质量织物纹理转移到任意形状的3D服装模型上,从而为3D虚拟试穿提供了更为真…

文章解读与仿真程序复现思路——电网技术EI\CSCD\北大核心《基于AGCN-LSTM模型的海上风电场功率概率预测 》

本专栏栏目提供文章与程序复现思路&#xff0c;具体已有的论文与论文源程序可翻阅本博主免费的专栏栏目《论文与完整程序》 论文与完整源程序_电网论文源程序的博客-CSDN博客https://blog.csdn.net/liang674027206/category_12531414.html 电网论文源程序-CSDN博客电网论文源…

端到端自动驾驶模型SparseDrive部署过程

SparseDrive 论文链接 https://arxiv.org/pdf/2405.19620 仓库链接 https://github.com/swc-17/SparseDrive 论文和模型的相关介绍大家可以参考其他博客的介绍&#xff0c;这里只介绍模型部署的过程和中间可能遇到的问题解决办法&#xff0c;以及代码解析和使用记录。 模型部署…

CyberRT通信介绍与基于Reader、Writer的通信实践(apollo9.0)

目录 数据通信场景 CyberRT中的通信方式 ​编辑 通信模式 话题通信 服务通信 参数通信 protobuf protobuf简介 protobuf文件编写 topic通信实验 实验环境 实验准备 代码编写 定义消息格式 发送消息 接收消息 定义编译规则 程序编译 运行程序 数据通信场景 …

fabric-sdk-go

Fabric-SDK-go 区块链网络搭建fabric-sdk代码代码结构&#xff1a;代码eg&#xff1a; 区块链网络搭建 使用fabric-sample的网络结构用容器搭建起测试网络即可。 fabric-sdk代码 代码很简易&#xff0c;主要为了了解怎么使用fabric为编程人员提供的sdk从而提供HTTP接口的情况…

浅谈华为 HarmonyOS Next

1. 万物互联时代的新机遇 随着万物互联时代的到来&#xff0c;智能应用从几十亿部手机扩展到数百亿个IoT设备&#xff0c;深刻改变了人们的生活方式。这为我们应用开发者带来了新的机遇和挑战。 机遇 : 目前正处于万物互联时代的前夕&#xff0c;正在经历手机单设备到全场景多…

技术分享:A-23OH型树脂在汽车涂装废溶剂回收中的应用

在当今汽车制造业竞争激烈的环境下&#xff0c;提高生产效率、降低成本的同时&#xff0c;满足环保要求已成为各制造商追求的核心目标。水性涂料因其环保、节能等多重优势&#xff0c;在汽车涂装领域的应用日益广泛。然而&#xff0c;随之而来的喷涂废溶剂处理问题也日益凸显。…

从 Hadoop 迁移到数据 Lakehouse 的架构师指南

从 Hadoop 到数据湖仓一体架构的演变代表了数据基础架构的重大飞跃。虽然 Hadoop 曾经以其强大的批处理能力统治着大数据领域&#xff0c;但如今的组织正在寻求更敏捷、更具成本效益和现代化的解决方案。尤其是当他们越来越多地开始实施 AI 计划时。根本没有办法让 Hadoop 为 A…

三周精通FastAPI:1 第一步入门

FastAPI是一个非常棒的python web和api框架&#xff0c;准备用三周的时间“精通它” 学习流程参考FastAPI官网的用户教程&#xff1a;教程 - 用户指南 - FastAPI 学前提示 运行代码 所有代码片段都可以复制后直接使用&#xff08;它们实际上是经过测试的 Python 文件&#x…