(每日一题) 力扣 283 移动零

文章目录

      • [**一、问题分析** 💡](#一、问题分析 💡)
      • [**二、算法思路** 🚀](#二、算法思路 🚀)
      • [**三、分步图解(以示例 `nums = [0,1,0,3,12]` 为例)** 📊](#三、分步图解(以示例 nums = [0,1,0,3,12] 为例) 📊)
        • [**1. 初始化指针**](#1. 初始化指针)
        • [**2. 第一次循环(`cur=0`)**](#2. 第一次循环(cur=0)
        • [**3. 第二次循环(`cur=1`)** 🎯](#3. 第二次循环(cur=1 🎯)
        • [**4. 第三次循环(`cur=2`)**](#4. 第三次循环(cur=2)
        • [**5. 第四次循环(`cur=3`)** 🔄](#5. 第四次循环(cur=3 🔄)
        • [**6. 第五次循环(`cur=4`)** ✨](#6. 第五次循环(cur=4 ✨)
      • [**四、代码解析** 🛠️](#四、代码解析 🛠️)
        • [**关键点解释** 📌](#关键点解释 📌)
      • [**五、总结** ✅](#五、总结 ✅)

一、问题分析 💡

题目要求将数组中的零元素全部移动到末尾,同时保持非零元素的原始顺序。关键约束是必须原地操作,即不能使用额外数组空间。这意味着我们需要在遍历过程中通过交换或覆盖元素实现目标。


二、算法思路 🚀

核心思想是使用双指针策略:

  1. 快指针 cur:从左到右遍历数组,寻找非零元素。
  2. 慢指针 dest:始终指向当前已处理好的非零序列的末尾下一个位置。

操作逻辑

  • nums[cur] 非零时,将其与 dest 位置交换,并让 dest 后移。
  • 最终,所有非零元素会被集中到数组前端,而 dest 之后的位置自然会被零填充。

三、分步图解(以示例 nums = [0,1,0,3,12] 为例) 📊

1. 初始化指针
  • dest = -1(表示尚未找到非零元素)

  • cur = 0(从数组头部开始遍历)

    初始状态:
    nums = [0, 1, 0, 3, 12]

    cur=0, dest=-1

2. 第一次循环(cur=0
  • nums[0] 是 0 → 不交换,cur 右移。

    状态不变:
    nums = [0, 1, 0, 3, 12]

    cur=1, dest=-1

3. 第二次循环(cur=1 🎯
  • nums[1] = 1 非零 → dest 先自增到 0,交换 nums[0]nums[1]

    交换后:
    nums = [1, 0, 0, 3, 12]

    cur=1, dest=0

  • cur 右移,dest 更新为 0。

4. 第三次循环(cur=2
  • nums[2] = 0 → 不交换,cur 右移。

    状态不变:
    nums = [1, 0, 0, 3, 12]

    cur=3, dest=0

5. 第四次循环(cur=3 🔄
  • nums[3] = 3 非零 → dest 自增到 1,交换 nums[1]nums[3]

    交换后:
    nums = [1, 3, 0, 0, 12]

    cur=3, dest=1

  • cur 右移,dest 更新为 1。

6. 第五次循环(cur=4
  • nums[4] = 12 非零 → dest 自增到 2,交换 nums[2]nums[4]

    交换后:
    nums = [1, 3, 12, 0, 0]

    cur=4, dest=2

  • 遍历结束,所有非零元素已按序排列在前,后续位置自动补零。


四、代码解析 🛠️

cpp 复制代码
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]);  // 🔄 交换到非零序列末尾
            }
        }
    }
};
关键点解释 📌
  1. dest 的初始化:初始为 -1,表示尚未找到任何非零元素。
  2. 交换条件 :仅当 nums[cur] 非零时,将其交换到 dest 的下一个位置(即 ++dest)。
  3. 时间复杂度:O(n) 🚀,仅一次遍历。
  4. 空间复杂度:O(1) 💾,原地操作。

五、总结

双指针法是解决此类问题的经典策略,通过一次遍历实现高效操作。核心在于维护两个指针的语义:

  • dest 始终指向已处理的非零序列末尾。
  • cur 负责探索未处理区域,将非零元素按序交换到前方。
相关推荐
xin007hoyo3 小时前
算法笔记.染色法判断二分图
数据结构·笔记·算法
mozun20203 小时前
VS BUG(6) LINK : fatal error LNK1158: 无法运行“rc.exe”
c++·bug·vs·链接器·资源文件
夜夜敲码3 小时前
C语言教程(十八):C 语言共用体详解
c语言·开发语言
安顾里4 小时前
Linux命令-iostat
linux·运维·服务器
whoarethenext4 小时前
初始https附带c/c++源码使用curl库调用
服务器·c++·qt·https·curl
100编程朱老师5 小时前
面试:什么叫Linux多路复用 ?
linux·运维·服务器
miracletiger5 小时前
uv 新的包管理工具总结
linux·人工智能·python
enyp805 小时前
麒麟系统(基于Ubuntu)上使用Qt编译时遇到“type_traits文件未找到”的错误
linux·qt·ubuntu
cloues break.5 小时前
C++进阶----多态
开发语言·c++
struggle20255 小时前
LinuxAgent开源程序是一款智能运维助手,通过接入 DeepSeek API 实现对 Linux 终端的自然语言控制,帮助用户更高效地进行系统运维工作
linux·运维·服务器·人工智能·自动化·deepseek