【优选算法】双指针算法:专题一

【优选算法】双指针算法:专题一

目录

引言:

【283.移动零】

1、题目描述

2、实现核心及思路

解题思路:

思路可视化:

代码实现:

代码测试:

【1089.复写零】

1、题目描述​

2、实现核心及思路

解题思路:

思路可视化:

代码实现:

代码测试:

【202. 快乐数】

1、题目描述

2、实现核心及思路

解题思路:

代码实现:

【11. 盛水最多容器】

1、题目描述

2、实现核心及思路

解题思路:

思路可视化:

代码实现:


引言:

常见的双指针有两种形式,一种是对撞指针,一种是快慢指针

对撞指针:一般用于顺序结构中,也称左右指针。

• 对撞指针从两端向中间移动。一个指针从最左端开始,另一个从最右端开始,然后逐渐往中间逼近。

• 对撞指针的终止条件一般是两个指针相遇或者错开(也可能在循环内部找到结果直接跳出循环),也就是:

◦ left == right (两个指针指向同一个位置)

◦ left > right (两个指针错开)
快慢指针:又称为龟兔赛跑算法,其基本思想就是使用两个移动速度不同的指针在数组或链表等序列 结构上移动。 如果我们要研究的问题出现循环往复的情况时,均可考虑使用快 慢指针的思想。

快慢指针的实现方式有很多种,最常用的一种就是:

• 在一次循环中,每次让慢的指针向后移动一位,而快的指针往后移动两位,实现一快一慢。
注意:这里的指针并不是C语言中学的 int*,char*的指针(地址),而是用变量记录数组下标(指针),变量++或--相当于指向 指针的移动。

【283.移动零】

1、题目描述

2、实现核心及思路

解题思路:
这里对撞指针(左右指针)的方案肯定是行不通的,因为题目要求移动 0 元素的同时还要保证非零元素的相对顺序。所以,我们采用两个指针都从数组的左端出发,把非零的元素依次找到放在前面,最后剩下 0 自然就排在数组后面了。

(1)刚开始让一个指针cur 先指向数组的第一个元素,另一个指针 dest 初始化为 -1。cur 用来找非零元素,dest 用来确定 cur 之前的 0 元素。

(2)当 cur 指向的元素为零则 cur++,继续向后找非零元素来将前面的 0 交换到后面;

(3)当 cur 指向的元素不为 0,则让 dest 向前移动一位(dest++),因为此时cur 之前的元素要么全为 0,要么dest 和 cur 指向同一个元素。然后,交换 cur 和 dest 指向的元素,cur 继续向后移动(cur++)

(4)结束条件:当 cur 走到数组尾部的时候意味着所有非零元素都找到了,结束。

思路可视化:
代码实现:
class Solution { public: void moveZeroes(vector<int>& nums) { int size=nums.size(); // 定义双指针 int dest=-1; int cur=0; // 指向数组第一个元素 while(cur<size) { // 找到非零元素 if(nums[cur]!=0) { dest++; swap(nums[dest],nums[cur]); } cur++; } } };
代码测试:
int main() { Solution s; vector<int> v1{ 1,0,2,0,3,0,4 }; vector<int> v2{ 0,0,0,6,5,4 }; s.moveZeroes(v1); for (auto e : v1) { cout << e << " "; } cout << endl; s.moveZeroes(v2); for (auto e : v2) { cout << e << " "; } return 0; }

【1089.复写零】

1、题目描述

2、实现核心及思路

解题思路:

解法一:额外创建一个等大的数组,然后从头开始遍历原数组,遇到非零元素正常写入新数组,遇到 0 元素,写入两次,当新数组写满就终止。

这种解法肯定是我们第一时间想到的,但题目明确限制在原数组上操作。

解法二:观察示例1可知,最终的输出结果即将原来的5 和 0 覆盖了,因为复写了两个 0 元素。所以,我们只需要将 [1,0,2,3,0,4]这部分数据按要求进行复写零。但如果,我们从前往后遍历进行复写零时就会出现覆盖 0 元素后一个元素的问题,所以,我们考虑从 4 开始向前遍历数组来复写零,并把遍历到的元素从原数组的最后依次往左边放。非零元素写一次,0 元素写两次。

⭐️⭐️所以,我们的问题就转化为了:先找最后一个有效元素(相当于上面的" 4 "),然后从最后一个有效元素开始从后往前遍历复写零。

步骤一:利用双指针法找最后一个有效元素位置,让 cur 指针开始指向第一个元素,然后遍历数组,同时定义 dest 指针(初始化为-1,即相当于指向数组起始位置的前一个位置)。当 cur 指向非零元素,left 向后移动一位,当 cur 指向零元素left 向后移动两位(即为复写零留出位置)。

结束条件:当 dest 指向的位置到数组的末尾时,即复写零后刚好能将数组写满。设数组大小为size,则dest < size - 1

细节一:



面对这种情形,即当进入循环后发现dest 移动两次后到了数组的末尾,此时,我们不能再让cur 向前移动了,如果再移动数组空间大小就不够复写两个 0 了。所以,在 cur++ 之前我们还要判断dest 是否已经 不满足 dest < size - 1了。

步骤二:继续利用双指针法,cur 指针已经指向了最后一个有效数据,所以让 dest 指向数组的最后。从 cur 指向的位置 开始向前遍历数组来复写零,当遍历到非零元素,dest 指向位置处仅将该元素写一次,然后 cur--,dest--,继续向前遍历;当遍历到 0 元素,dest 指向位置处先写一次 0 ,然后dest向左移动(dest--),并再写一次 0,然后 cur--,dest-- ...  以此类推。

结束条件:当 cur 遍历到数组的起始位置,即复写结束。cur >= 0

细节二:



当cur 指向最后一个有效数据且为 0 时,dest 需要向后移动两次,但是数组空间大小只剩下一个位置,即dest 出了数组。意味着我们在先前复写零时,cur 指向的这个 0 元素只能复写一次,所以,我们需要单独处理。
思路可视化:
代码实现:
class Solution { public: void duplicateZeros(vector<int>& arr) { int cur = 0; // 指向数组首元素 int dest = -1; // 指向数组开头之前 int size = arr.size(); while (dest < size - 1) // 结束条件:dest到数组末尾或之后 { if (arr[cur] == 0) dest += 2; // 遍历到 0 元素,dest 向后移动两位 else dest++; // 非零元素,dest 正常向后移动一位 if (dest < size - 1) cur++; // 细节一 } if (dest >= size) // 细节二:处理最后一个 0 元素只复写一次 { arr[size - 1] = 0; dest = size - 2; cur--; } while (cur >= 0) // 结束条件 { if (arr[cur] == 0) // 复写零 { arr[dest--] = 0; arr[dest--] = 0; } else arr[dest--] = arr[cur]; cur--; // 向前遍历 } } };
代码测试:
int main() { Solution s; vector<int> v1{ 1, 0, 2, 3, 0, 4, 5, 0 }; // 正常情况 vector<int> v2{ 1, 0, 2, 3, 0, 4, 5 }; // 细节一 vector<int> v3{ 1, 0, 2, 3, 0, 4 }; // 细节二 s.duplicateZeros(v1); s.duplicateZeros(v2); s.duplicateZeros(v3); for (auto e : v1) cout << e << " "; cout << endl; for (auto e : v2) cout << e << " "; cout << endl; for (auto e : v3) cout << e << " "; return 0; }

【202.快乐数】

1、题目描述

2、实现核心及思路

解题思路:

对于n = 19,经过求和后,最终结果变为了1,即19为快乐数;n = 2,经过求和,最终又回到了起点。

结论:对于每个非零整数,进行上面的操作,要么最终得到1,要么继续回到一个求和过程中的某一个值(非1),但不一定回到最开始(像2这样)。

即经过不断的求和(按上面的规则),最会进入到一个循环当中,得到1 的这种情况也不例外,1求和还是1,也是循环。而对于这种循环的问题:我们均可以用快慢指针的方法解决。

假设慢指针一次走一步,快指针一次走两步,那么快指针一定先入环,但由于两者的速度差,最终肯定会相遇(结束条件)。

这里我们先不做证明!!!(后面会补充)

对于这个题而言:慢指针走一步即对应一次求和一次,快指针走两步即对应连续求和两次。

结束条件:当快慢指针相遇

然后判断:如果相遇时,快慢指针指向的和为 1 ,这个数即为快乐数;反之,则不是。

代码实现:
class Solution { public: // 求和函数 int Sum(int n) { int sum = 0; while(n) { int a = n%10; sum += pow(a,2); n/=10; } return sum; } bool isHappy(int n) { int slow = Sum(n); // 慢指针 int fast = Sum(n); // 快指针 fast = Sum(fast); while(slow != fast) // 结束条件 { slow = Sum(slow); fast = Sum(fast); fast = Sum(fast); } if(slow == 1) return true; else return false; } };

【11​​​​​​.盛水最多容器】

1、题目描述

2、实现核心及思路

解题思路:

解法一:暴力解法,两次for循环,算出所有的体积,找出最大值。

解法二:对撞指针(左右指针),left 和 right 指针向右向左找符合的桶(区间)。主要是依靠单调性,当left 和 right 向内移动时,宽度一定是变短的,体积 V = 宽度 * 高,所以要想找到最大的体积,只能依靠找到更大的高度。

所以,让left 和 right 中指向高度较小的一个向内移动, 找更大的高度,才有可能使体积更大。

同时,每次计算一个体积,通过比较获得最大体积。

💡库当中有一个 max 函数可以帮我们找两个数中的较大值,头文件<algorithm>
思路可视化:
代码实现:
class Solution { public: int maxArea(vector<int>& height) { int left = 0; // 左指针 int right = height.size() - 1; // 右指针 int m = 0; while(left < right) { int h = height[left] > height[right] ? height[right] : height[left]; // 找较小边 int V = (right - left) * h; // 计算体积 m = max(m,V); // 更新体积的值 if(height[left] < height[right]) left++; else right--; } return m; } };

Read more

【OpenClaw从入门到精通】第10篇:OpenClaw生产环境部署全攻略:性能优化+安全加固+监控运维(2026实测版)

【OpenClaw从入门到精通】第10篇:OpenClaw生产环境部署全攻略:性能优化+安全加固+监控运维(2026实测版)

摘要:本文聚焦OpenClaw从测试环境走向生产环境的核心痛点,围绕“性能优化、安全加固、监控运维”三大维度展开实操讲解。先明确生产环境硬件/系统选型标准,再通过硬件层资源管控、模型调度策略、缓存优化等手段提升响应速度(实测响应效率提升50%+);接着从网络、权限、数据三层构建安全防护体系,集成火山引擎安全方案拦截高危操作;最后落地TenacitOS可视化监控与Prometheus告警体系,配套完整故障排查清单和虚拟实战案例。全文所有配置、代码均经实测验证,兼顾新手入门实操性和进阶读者的生产级部署需求,帮助开发者真正实现OpenClaw从“能用”到“放心用”的跨越。 优质专栏欢迎订阅! 【DeepSeek深度应用】【Python高阶开发:AI自动化与数据工程实战】【YOLOv11工业级实战】 【机器视觉:C# + HALCON】【大模型微调实战:平民级微调技术全解】 【人工智能之深度学习】【AI 赋能:Python 人工智能应用实战】【数字孪生与仿真技术实战指南】 【AI工程化落地与YOLOv8/v9实战】【C#工业上位机高级应用:高并发通信+性能优化】 【Java生产级避坑指南:

By Ne0inhk
ARM Linux 驱动开发篇--- Linux 并发与竞争实验(互斥体实现 LED 设备互斥访问)--- Ubuntu20.04互斥体实验

ARM Linux 驱动开发篇--- Linux 并发与竞争实验(互斥体实现 LED 设备互斥访问)--- Ubuntu20.04互斥体实验

🎬 渡水无言:个人主页渡水无言 ❄专栏传送门: 《linux专栏》《嵌入式linux驱动开发》《linux系统移植专栏》 ❄专栏传送门: 《freertos专栏》《STM32 HAL库专栏》 ⭐️流水不争先,争的是滔滔不绝  📚博主简介:第二十届中国研究生电子设计竞赛全国二等奖 |国家奖学金 | 省级三好学生 | 省级优秀毕业生获得者 | ZEEKLOG新星杯TOP18 | 半导纵横专栏博主 | 211在读研究生 在这里主要分享自己学习的linux嵌入式领域知识;有分享错误或者不足的地方欢迎大佬指导,也欢迎各位大佬互相三连 目录 前言  一、实验基础说明 1.1、互斥体简介 1.2 本次实验设计思路 二、硬件原理分析(看过之前博客的可以忽略) 三、实验程序编写 3.1 互斥体 LED 驱动代码(mutex.c) 3.2.1、设备结构体定义(28-39

By Ne0inhk
Flutter for OpenHarmony:swagger_dart_code_generator 接口代码自动化生成的救星(OpenAPI/Swagger) 深度解析与鸿蒙适配指南

Flutter for OpenHarmony:swagger_dart_code_generator 接口代码自动化生成的救星(OpenAPI/Swagger) 深度解析与鸿蒙适配指南

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net 前言 后端工程师扔给你一个 Swagger (OpenAPI) 文档地址,你会怎么做? 1. 对着文档,手写 Dart Model 类(容易写错字段类型)。 2. 手写 Retrofit/Dio 的 API 接口定义(容易拼错 URL)。 3. 当后端修改了字段名,你对着报错修半天。 这是重复劳动的地狱。 swagger_dart_code_generator 可以将 Swagger (JSON/YAML) 文件直接转换为高质量的 Dart 代码,包括: * Model 类:支持 json_serializable,带 fromJson/

By Ne0inhk
Linux 开发别再卡壳!makefile/git/gdb 全流程实操 + 作业解析,新手看完直接用----《Hello Linux!》(5)

Linux 开发别再卡壳!makefile/git/gdb 全流程实操 + 作业解析,新手看完直接用----《Hello Linux!》(5)

文章目录 * 前言 * make/makefile * 文件的三个时间 * Linux第一个小程序-进度条 * 回车和换行 * 缓冲区 * 程序的代码展示 * git指令 * 关于gitee * Linux调试器-gdb使用 * 作业部分 前言 做 Linux 开发时,你是不是也遇到过这些 “卡脖子” 时刻?写 makefile 时,明明语法没错却报错,最后发现是依赖方法行没加 Tab;想提交代码到 gitee,记不清 git add/commit/push 的 “三板斧”,还得反复搜教程;用 gdb 调试程序,输了命令没反应,才想起编译时没加-g生成 debug 版本;甚至连写个进度条,都搞不懂\r和\n的区别,导致进度条乱跳…… 其实这些问题,

By Ne0inhk