《算法题讲解指南:优选算法-双指针》--01移动零,02复写零

《算法题讲解指南:优选算法-双指针》--01移动零,02复写零

🔥小叶-duck个人主页

❄️个人专栏《Data-Structure-Learning》

《C++入门到进阶&自我学习过程记录》

未择之路,不须回头
已择之路,纵是荆棘遍野,亦作花海遨游


目录

一、双指针算法介绍

  1、对撞指针

  2、快慢指针

01、移动零

题目链接:

题目描述:

题目示例:

算法思路:

算法流程:

C++ 代码演示:

算法总结:

02、复写零

题目链接:

题目描述:

题目示例:

算法思路:

算法流程:

C++代码演示:

算法总结及流程解析:

结束语


一、双指针算法介绍

      在正式讲解本次的算法题之前我们先来看看算法中一个非常常用的方法——双指针。双指针有两种形式,一种对撞指针,一种是左右指针。

  1、对撞指针

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

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

  2、快慢指针

又称龟兔赛跑算法,其基本思想就是使用两个移动速度不同的指针在数组或链表等序列结构上移动。

      这种方法对于处理环形链表或数组非常有用。其实不单单是环形链表或者数组,如果我们要研究的问题出现循环往复的情况时,均可考虑使用快慢指针的思想。

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

  • 在一次循环中,每次让慢的指针向后移动一位,而快的指针往后移动两位,实现一快一慢

01、移动零

题目链接:

283. 移动零 - 力扣(LeetCode)

题目描述:

题目示例:

解法:(快排的思想:数组划分区间-数组分块)

      【数组分块】是非常常见的一种算法技巧,主要就是根据一种划分方式,将数组的内容分成左右两部分。这种类型的题,一般就是使用【双指针】来解决。

算法思路:
  • 我们可以用一个 cur 指针来扫描整个数组,另一个 dest 指针用来记录非零序列的最后一个位置,根据 cur 在扫描的过程中,遇到的不同情况,分类处理,实现数组的划分。
  • 在 cur 遍历期间,保证【0,dest】区间的元素全部都是非零元素,【dest+1,cur-1】区间的元素全是零,而 cur 后面的元素则是未处理的。
算法流程:

  1、初始化 cur = 0(用来遍历数组),dest = -1(指向非零元素序列的最后一个位置。因为刚开始还没有进行遍历,此时相当于还没有非零元素的序列,因此初始化为 -1

  2.cur 依次往后遍历每个元素,遍历到的元素会有下面两种情况:
    2.1 遇到的元素是 0 ,cur直接++ ,不需要对 dest 进行操作。因为我们的目标是让【dest+1,cur-1】内的元素全都是0,因此当 cur  遇到 0 的时候,直接 ++ ,就可以保证【dest+1,cur-1】这个区间内依然全为0;
    2.2 遇到的元素不是 dest++,并且交换 cur 位置和 dest 位置的元素,之后让 cur++,扫描下一个元素。

  • 因为 dest 指向的位置是非零元素区间的最后一个位置,如果扫描到一个新的非零元素,那么这个非零元素的位置应该在 dest+1 的位置上,因此 dest 先自增 1
  • dest++ 之后,指向的元素就是 元素(因为非零元素区间末尾的后一个元素(下标为 dest+1)就是 ),因此可以交换到 cur 所处的位置上,实现【0,dest】的元素全部都是非零元素,【dest+1,cur-1】的元素全是零。

C++ 代码演示:

class Solution { public: void moveZeroes(vector<int>& nums) { int dest = -1; int cur = 0; while(cur != nums.size()) { // if(nums[cur] == 0) // { // cur++; // } // else // { // swap(nums[++dest], nums[cur++]); // } //代码优化 if(nums[cur])//if判断为假则说明cur遇到0 { swap(nums[++dest], nums[cur]); } cur++; } } };

算法总结:

      这里用到的方法就是我们在数据结构中学习 【快排算法】的时候,【数据划分】过程的重要一步。如果将快排算法的递归拆解成单趟的话,这一小段代码就是实现快排单趟的【核心步骤】。

02、复写零

题目链接:

1089. 复写零 - 力扣(LeetCode)

题目描述:

题目示例:

解法:(原地复写——双指针)

算法思路:
  • 如果【从前往后】进行原地复写操作的话,由于 0 的出现会复写两次,导致没有复写的数【被覆盖掉】。因此我们选择【从后往前】的复写策略。
  • 但是 【从后往前】复写的时候,我们需要找到 【最后一个复写的数】,因此我们的大体流程分两步:1.先找到最后一个复写的数;2.然后从后往前进行复写操作
算法流程:

  1.初始化两个指针 cur = 0dest = -1

  2.找到最后一个复写的数:

  • 判断 cur 位置的元素:(1)如果是 0 的话,dest 往后移动两位;(2)否则,dest 往后移动一位
  • 判断 dest 这时候是否已经到结束位置,如果结束就终止循环;
  • 如果没有结束,cur++,继续判断。

  3.判断 dest 是否越界到 的位置:

如果越界,执行下面三步:

  • n-1 位置的值修改成 0 ;
  • cur 向前移动一步(cur--)
  • dest 向前移动两步(dest -= 2);

  4.从 cur 位置开始往前遍历原数组,依次还原出复写后的结果数组:

    4.1 判断 cur 位置的值:

  • 如果是 0 :dest 以及 dest-1 位置修改成 dest-=2
  • 如果非零:dest 位置修改成 0 ,dest -= 1

    4.2 cur--,复写下一个位置。

C++代码演示:

class Solution { public: void duplicateZeros(vector<int>& arr) { // 1. 先找到最后⼀个复写的数 int dest = -1; int cur = 0; while(1) { dest++; if(arr[cur] == 0) { dest++; } if(dest >= arr.size() - 1) { break; } cur++; } // 2. 处理越界情况 if(dest == arr.size()) { arr[arr.size() - 1] = 0; dest -= 2; cur--; }//之所以会越界是因为如果最后一个复写的数为0 //则可能出现复写的位置在下标n-1和n,但下标为n就是越界 //也就是说实际上并没有复写两遍0而只是数组最后一个位置复写为0 //所以如果越界操作就是数组最后位置手动置为0后让dest回到倒数第二个位置 //cur--就是让最后一个复写的数变成前一个 // 3. 从后向前完成复写操作 while(cur >= 0) { if(arr[cur] == 0) { arr[dest--] = arr[cur]; } arr[dest--] = arr[cur--]; } } };

算法总结及流程解析:

结束语

      到此,本次的算法题就讲解完了,这是小萌新讲解算法题的开始,后面我会不断地对非常经典且精选的算法题为大家进行讲解,希望大家能有所收获!

Read more

Realtek 8922AE Ubuntu22.04 无WiFi驱动解决

Realtek 8922AE,在下载驱动的时候可能有前缀rtw、RTL,具体我没太区分,尝试了好几个版本,现记录最傻瓜式的安装方式(如果是双系统,驱动的型号在Windows下高级网络设置-硬件和连接属性中可以看,在ubuntu中,本人设备是拯救者7000p,使用 lspci | grep -i Network 可看) 注意,确保你先使用某种方式能连上网(有线\手机USB) 首先需要设定BIOS的Secure Boot 为disabled 可通过指令查询: mokutil --sb-state 结果为SecureBoot disabled即可,否则先关机设置,拯救者是按F2进BIOS,备选方案是使用以下指令,将驱动对应的公钥(mok.pub)导入到系统中。导入后,后续还需要在计算机重启时进入 “Machine Owner Key (MOK)” 管理界面,选择注册刚刚导入的密钥,完成签名验证流程: sudo mokutil --import /var/lib/

By Ne0inhk
Flutter 三方库 frontend_server_client 深入鸿蒙编译后端极限热接驳管线适配:以桥连中继重组增量渲染图节点并强推微秒级热重载指令下发-适配鸿蒙 HarmonyOS ohos

Flutter 三方库 frontend_server_client 深入鸿蒙编译后端极限热接驳管线适配:以桥连中继重组增量渲染图节点并强推微秒级热重载指令下发-适配鸿蒙 HarmonyOS ohos

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 三方库 frontend_server_client 深入鸿蒙编译后端极限热接驳管线适配:以桥连中继重组增量渲染图节点并强推微秒级热重载指令下发提振终端即时可视效能 在鸿蒙应用的跨平台引擎开发、自定义开发工具链或高频的热重载(Hot Reload)性能优化设计中,如何精准地控制 Dart 源码到 Kernel 文件(.dill)的编译转换过程?frontend_server_client 提供了一套与 Dart Frontend Server 通信的标准化客户端。本文将详解该库在 OpenHarmony 上的适配要点。 前言 什么是 frontend_server_client?它是 Dart SDK 中前端编译服务的封装包。它通过特定的交互协议,允许外部程序(如 IDE 插件或自定义 CLI

By Ne0inhk
Flutter 组件 flutter_sheet_localization 的适配 鸿蒙Harmony 实战 - 驾驭云端词典自动化、实现鸿蒙端国际化词条无感更新与多语言 Key 生成方案

Flutter 组件 flutter_sheet_localization 的适配 鸿蒙Harmony 实战 - 驾驭云端词典自动化、实现鸿蒙端国际化词条无感更新与多语言 Key 生成方案

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 组件 flutter_sheet_localization 的适配 鸿蒙Harmony 实战 - 驾驭云端词典自动化、实现鸿蒙端国际化词条无感更新与多语言 Key 生成方案 前言 在鸿蒙(OpenHarmony)生态的全球化应用开发中,面对上百个涉及金融支付、法律协议以及动态营销文案的多语言(i18n)词条映射。如果仅仅依靠传统的本地 intl 方案 手动修改 .arb 或 .json 文件。那么不仅会导致开发与翻译团队之间的“沟通断层”。更会因为频繁的手动拷贝错误引发严重的生产事故。 我们需要一种“云端协同、本地免维护”的翻译生产艺术。 flutter_sheet_localization 是一套专注于将 Google Sheets(或是兼容的 CSV 系统)

By Ne0inhk
Flutter 三方库 front_end — 揭秘鸿蒙应用编译过程中的源码解析与内核转换内幕,实现鸿蒙深度适配下的自研工具链内核实战全解(适配鸿蒙 HarmonyOS Next ohos)

Flutter 三方库 front_end — 揭秘鸿蒙应用编译过程中的源码解析与内核转换内幕,实现鸿蒙深度适配下的自研工具链内核实战全解(适配鸿蒙 HarmonyOS Next ohos)

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net。 Flutter 三方库 front_end — 揭秘鸿蒙应用编译过程中的源码解析与内核转换内幕,实现鸿蒙深度适配下的自研工具链内核实战全解 前言 当你点击 DevEco Studio 的“运行”按钮,将 Flutter 代码部署到鸿蒙(OpenHarmony)真机时,幕后发生了一场复杂的“代码进化”。在这场进化的第一阶段,源代码需要经历解析、语义分析,并转变为一种中间表示格式(Kernel)。 front_end (通常指 package:front_end 或相关的 CFE - Common Front End) 是 Dart 编译器架构中掌管“解析与验证”的核心大脑。在 Flutter for

By Ne0inhk