算法每日双题精讲——双指针(移动零,复写零)

news2025/1/10 21:55:25

🌟快来参与讨论💬,点赞👍、收藏⭐、分享📤,共创活力社区。 🌟

别再犹豫了!快来订阅我们的算法每日双题精讲专栏,一起踏上算法学习的精彩之旅吧!💪


💯前言

在算法的世界里,双指针技巧常常能发挥出神奇的作用😎。今天,我们就来精讲两道利用双指针解决的经典题目:移动零和复写零。 

📣由于俩道题目均为数组,这里的双指针算法指的是:利用数组下标代替指针

当我们遇到,数组分块,数组划分的问题时,可以考虑使用双指针法。 

 


 💯双指针的作用

✍两个指针的作用:

  • cur:从左往右扫描数组,遍历数组
  • dest:已处理的区间内,非零元素的最后一个位置

分为三个区间:

  1. [0, dest]
  2. [dest + 1,cur -1]
  3. [cur, n -1]

💯移动零

题目链接👉【力扣】 

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

示例:
输入:nums = {0,1,0,3,12}
输出:{1,3,12,0,0}

⭐解题思路:

我们可以使用双指针法来解决这个问题。一个指针 cur用于遍历整个数组,另一个指针dest 用于指向当前非零元素应该放置的位置。当遇到非零元素时,将其放置在dest 指针所指的位置,并将dest 指针向后移动一位。遍历结束后,从dest 指针开始到数组末尾的位置全部设置为零。

 😀俩个指针将数组分为三个区间:

  1. [0, dest]:全是非0的元素(已经处理)
  2. [dest + 1,cur -1]:都是0(已经处理)
  3. [cur, n -1]:还未处理过的

 

 代码实现(以 C++ 为例):
class Solution {
public:
    void moveZeroes(vector<int>& nums) {
        // dest 用于标记已处理的非零元素的最后位置
        int dest = -1;
        // cur 用于遍历整个向量
        int cur = 0;
        while (cur < nums.size()) {
            // 如果当前位置的元素为 0
            if (nums[cur] == 0) {
                cur++;
            } else {
                // 先将 dest 加 1,标记下一个非零元素应放置的位置
                swap(nums[++dest], nums[cur]);
                cur++;
            }
        }
    }
};
👀复杂度分析:
  • 时间复杂度:O(n),其中 n 是数组的长度。我们只需要遍历一次数组。
  • 空间复杂度:O(1),只使用了有限的额外空间。

💯复写零

题目链接👉【力扣】

题目描述:给你一个长度固定的整数数组 arr,请你将该数组中出现的每个零都复写一遍,并将其余的元素向右平移。注意❗:不要在超过该数组长度的位置写入元素。

示例:
输入:arr = {1,0,2,3,0,4,5,0}
输出:{1,0,0,2,3,0,0,4}

 

解题思路:

同样可以使用双指针法来解决这个问题。一个指针cur用于遍历数组,另一个指针dest用于指向复写零后数组中元素应该放置的位置。当遇到零元素时,将dest指针后的元素依次向后移动两位,并在dest和 dest+1 的位置都放置零。当遇到非零元素时,将其放置在dest指针所指的位置,并将dest指针向后移动一位。

🙋这个解题思路是怎么来的呢? 
  1. 首先我们从左往右遍历数组, 

     

    当arr[cur]!=0时,我们让dest的后面一个的值赋予a[cur]正指向的那个值

    当arr[cur]==0时,我们让dest的后俩个值都赋予0

    当走到这一步时:


    cur找不到下一个为2的值了,因此我们不能从左往右遍
  2. 我们从右往左遍历,dset指向最后一个元素,cur指向最后一个要复写的数
    当a[cur]!=0时,让a[dest]=a[cur],然后cur--,dest--;
    当a[cur]==0时,让a[dest--]=0,a[dest--]=0,cur--;
    这样遍历没有问题,因此我们选择从右往左遍历
    所以我们只要找到最后要“复写”的数即可

⭐找最后一个要复写的数 

 

👇起初让cur指向数组的开头,dest指向-1的位置: 

代码实现(以 C++ 为例): 
class Solution {
public:
    void duplicateZeros(vector<int>& arr) {
        // dest 用于标记复制零元素后的新位置,初始值为 -1
        int dest = -1;
        // cur 用于遍历原始数组,初始值为 0
        int cur = 0;
        int n = arr.size();
        // 遍历原始数组,确定复制零元素后的新位置
        while (cur < n) {
            // 如果当前元素不为 0
            if (arr[cur]!= 0) {
                // dest 向后移动一位
                dest++;
            } else {
                // 如果当前元素为 0,dest 向后移动两位(因为要复制一个零)
                dest += 2;
            }
            // 如果 dest 已经到达或超过新数组的最后一个位置,跳出循环
            if (dest >= n - 1) break;
            // cur 向后移动一位,继续遍历原始数组
            cur++;
        }
        // 如果 dest 正好等于新数组的长度
        if (dest == n) {
            // 将新数组的最后一个位置设为 0
            arr[n - 1] = 0;
            // dest 回退两位
            dest -= 2;
            // cur 回退一位,因为上一步 cur 多走了一步
            cur--;
        }
        // 从后往前遍历原始数组,进行复制操作
        while (cur >= 0) {
            // 如果当前元素不为 0
            if (arr[cur]!= 0) {
                // 将当前元素复制到新位置
                arr[dest] = arr[cur];
                // cur 和 dest 都向前移动一位
                cur--;
                dest--;
            } else {
                // 如果当前元素为 0,先将 0 复制到 dest 位置,再将另一个 0 复制到 dest - 1 位置
                arr[dest--] = 0;
                arr[dest--] = 0;
                // cur 向前移动一位
                cur--;
            }
        }
    }
};
👀复杂度分析:
  • 时间复杂度:O(n),其中 n 是数组的长度。我们需要遍历两次数组。
  • 空间复杂度:O(1),只使用了有限的额外空间。

💯总结

通过这两道题目,我们可以看到双指针算法在处理数组相关问题时的高效性和灵活性👏。希望大家在今后的算法学习中,能够熟练掌握双指针技巧,解决更多复杂的问题💪。


我以后还会对 算法 相关知识进行更多的创作,欢迎大家关注我,一起探索 算法 的奇妙世界😜

👉【A Charmer】

 

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

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

相关文章

【Android】View—基础知识,滑动,弹性滑动

基础知识 什么是View 在 Android 中&#xff0c;View 是用户界面&#xff08;UI&#xff09;中的基本组件&#xff0c;用于绘制图形和处理用户交互。所有的 UI 组件&#xff08;如按钮、文本框、图片等&#xff09;都是 View 的子类。可以说&#xff0c;View 是构建 Android …

【Unity】Game Framework框架学习使用

前言 之前用过一段时间的Game Framework框架&#xff0c;后来有那么一段时间都做定制小软件&#xff0c;框架就没再怎么使用了。 现在要做大型项目了&#xff0c;感觉还是用框架好一些。于是又把Game Framework拾起来了。 这篇文章主要是讲Game Framework这个框架是怎么用的…

【SoC设计指南 基于Arm Cortex-M】学习笔记1——AMBA

AMBA简介 先进微控制器总线架构&#xff08;Advanced Microcontroller Bus Architecture&#xff0c;AMBA&#xff09;是用在arm处理器上的片上总线协议规范集。 AMBA总线协议规范集包含AHB、APB、AXI等。 AHB&#xff1a;先进高性能总线(Advanced High-performance Bus) APB&…

pytorch模型转onnx的动态batch转换说明

将PyTorch模型&#xff08;.pth&#xff09;转换为ONNX格式时&#xff0c;通常需要指定一个batch size。这是因为ONNX模型需要一个固定的输入形状&#xff0c;而批处理大小是输入形状的一部分。 下面是一个简单的转换示例&#xff0c;假设你已经加载了一个PyTorch模型&#xff…

【王木头】最大似然估计、最大后验估计

目录 一、最大似然估计&#xff08;MLE&#xff09; 二、最大后验估计&#xff08;MAP&#xff09; 三、MLE 和 MAP 的本质区别 四、当先验是均匀分布时&#xff0c;MLE 和 MAP 等价 五、总结 本文理论参考王木头的视频&#xff1a; 贝叶斯解释“L1和L2正则化”&#xff…

从0到1基于LangChain制作一个AI猫娘

前言&#xff1a; 看到B站上的AIVtuber的项目落地了&#xff0c;就心血来潮想制作一个AI的猫娘供自己使用&#xff0c;顺便出一个简单的教程&#xff0c;跳过理论&#xff0c;直接实践&#xff0c;作者也还在学习摸索中&#xff0c;所以有错误可以直接在评论区指正。&#xff0…

Vue全栈开发旅游网项目(7)-搜索界面开发及其接口联调

1.搜索界面开发 1.1 模糊查询 文件地址&#xff1a;pycharm- class SightListView(ListView):paginate_by 5def get_queryset(self):#is_validTrue&#xff1a;表中is_valid列&#xff0c;有值则被查询出来query Q(is_validTrue)#1.获得热门景点is_hot self.request.GET.…

python识别ocr 图片和pdf文件

#识别图片 pip3 install paddleocr pip3 install paddlepaddle#识别pdf pip3 install PyMuPDF 重点&#xff1a;路径不能有中文&#xff0c;不然pdf文件访问不了 from paddleocr import PaddleOCR from rest_framework.response import Response from rest_framework.views im…

量化分析工具日常操作日记-5-通合科技

使用量化分析微信小程序工具“梦想兔企业智能风险分析助手”日常操作日记-5-军工-通合科技&#xff08;300491&#xff09;。 周末国家新政策&#xff0c;要大力支持军工行业&#xff0c;我用工具挖掘了两个低位股&#xff0c;供大家参考。通合科技&#xff08;300491&#xff…

详解基于C#开发Windows API的SendMessage方法的鼠标键盘消息发送

在C#中&#xff0c;SendMessage方法是一个强大的工具&#xff0c;它允许我们与Windows API交互&#xff0c;模拟键盘和鼠标事件。本文将详细介绍如何使用SendMessage方法来发送鼠标和键盘消息。 1. SendMessage方法概述 SendMessage是Windows API中的一个函数&#xff0c;它用…

CSS教程(三)- CSS 三大特性

1. 层叠性 介绍 多组CSS样式共同作用于一个元素&#xff0c;就会出现 覆盖&#xff08;层叠&#xff09; 另一个冲突的样式。 层叠原则 样式冲突&#xff1a;遵循就近原则&#xff08;哪个样式离结构近&#xff0c;就执行哪个样式&#xff09; 样式不冲突&#xff0c;就不会重…

CyclicBarrier使用详解及遇到的坑

上一篇文章讲的是关于是使用CountDownLatch实现生成年底报告遇到的问题&#xff0c;这个计数器和CyclicBarrier也有类似功能&#xff0c;但是应用场景不同。 一、应用场景 CountDownLatch&#xff1a; 有ABCD四个任务&#xff0c;ABC是并行执行,等ABC三个任务都执行完…

Java-I/O框架14:Properties集合及使用

视频链接&#xff1a;16.32 Properties使用&#xff08;2&#xff09;_哔哩哔哩_bilibilihttps://www.bilibili.com/video/BV1Tz4y1X7H7?spm_id_from333.788.player.switch&vd_sourceb5775c3a4ea16a5306db9c7c1c1486b5&p32 1.Properties集合 特性&#xff1a; 存储…

Windows下mysql数据库备份策略

Windows下mysql的增量备份和全量备份&#xff0c;并利用schtasks设置定时任务执行bat脚本。 一、备份要求 序号 备份类型 备份频次 备份时间 1 增量备份 每周一-每周六各一次 18:00:00 2 全量备份 每周日一次 18:00:00 二、备份方法 2.1增量备份 2.1.1准备工作…

架构师备考-概念背诵(软件工程)

软件工程 软件开发生命周期: 软件定义时期:包括可行性研究和详细需求分析过程,任务是确定软件开发工程必须完成的总目标,具体可分成问题定义、可行性研究、需求分析等。软件开发时期:就是软件的设计与实现,可分成概要设计、详细设计、编码、测试等。软件运行和维护:就是…

【Linux】Linux入门实操——vim、目录结构、远程登录、重启注销

一、Linux 概述 1. 应用领域 服务器领域 linux在服务器领域是最强的&#xff0c;因为它免费、开源、稳定。 嵌入式领域 它的内核最小可以达到几百KB, 可根据需求对软件剪裁&#xff0c;近些年在嵌入式领域得到了很大的应用。 主要应用&#xff1a;机顶盒、数字电视、网络…

【Java项目】基于SpringBoot的【生鲜交易系统】

技术简介&#xff1a; 系统软件架构选择B/S模式、java技术和MySQL数据库等&#xff0c;总体功能模块运用自顶向下的分层思想。 系统简介&#xff1a; 考虑到实际生活中在生鲜交易方面的需要以及对该系统认真的分析,将系统权限按管理员&#xff0c;用户这两类涉及用户划分。 (…

AI Weekly『11月4-10日』: Anthropic发布Claude 3.5 Haiku,腾讯开源混元-Large模型!

大家好&#xff0c;我是木易&#xff0c;一个持续关注AI领域的互联网技术产品经理&#xff0c;国内Top2本科&#xff0c;美国Top10 CS研究生&#xff0c;MBA。我坚信AI是普通人变强的“外挂”&#xff0c;专注于分享AI全维度知识&#xff0c;包括但不限于AI科普&#xff0c;AI工…

贪心算法day3(最长递增序列问题)

目录 1.最长递增三元子序列 2.最长连续递增序列 1.最长递增三元子序列 题目链接&#xff1a;. - 力扣&#xff08;LeetCode&#xff09; 思路&#xff1a;我们只需要设置两个数进行比较就好。设a为nums[0]&#xff0c;b 为一个无穷大的数&#xff0c;只要有比a小的数字就赋值…

vue实现图片无限滚动播放

本人vue新手菜鸡&#xff0c;文章为自己在项目中遇到问题的记录&#xff0c;如有不足还请大佬指正 文章目录 实现效果代码展示总结 因为刚接触vue&#xff0c;本想着看看能不能用一些element的组件实现图片的轮播效果&#xff0c;尝试使用过element-UI里的走马灯Carouse&#x…