LeetCode 283:原地移动零的优雅解法

LeetCode 热题 283. 移动零:原地操作的优雅解法

一、题目理解:需求很明确

题目要求:

  • 把数组里的所有0移到末尾
  • 保持非零元素的相对顺序
  • 必须原地操作(不能复制新数组)

比如输入[0,1,0,3,12],输出得是[1,3,12,0,0];如果输入本身就是[0],直接返回[0]就行。

二、我的解题思路:两步走策略

一开始我想的是「双指针交换」,但后来发现更简洁的是 **"先填非零,再补零"** 的思路:

解法核心逻辑

  1. 第一步:把非零元素 "往前搬" 用一个变量stack_size记录 "非零元素应该放的位置",遍历数组时,遇到非零元素就放到nums[stack_size],然后stack_size自增。(相当于把所有非零元素按顺序 "堆" 在数组前半部分)

  2. 第二步:给剩下的位置补 0 遍历结束后,从stack_size的位置开始,把数组剩下的部分全填成 0。

代码实现(C++)

cpp

复制代码
class Solution {
public:
    void moveZeroes(vector<int>& nums) {
        int stack_size = 0;
        // 第一步:填充非零元素
        for (int x : nums) {
            if (x) {
                nums[stack_size++] = x;
            }
        }
        // 第二步:补0
        fill(nums.begin() + stack_size, nums.end(), 0);
    }
};

三、为什么这个解法高效?

  • 时间复杂度 O (n):只遍历数组两次(一次填非零,一次补零),n 是数组长度。
  • 空间复杂度 O (1):完全原地操作,没额外开数组。
  • 操作次数少:比起 "遇到 0 就交换" 的写法,减少了不必要的交换操作(比如连续多个 0 的情况)。

四、拓展:双指针交换法(另一种思路)

其实也可以用双指针 实现(比如left指针指向 "下一个非零元素该放的位置"):

cpp

复制代码
void moveZeroes(vector<int>& nums) {
    int left = 0;
    for (int i = 0; i < nums.size(); i++) {
        if (nums[i] != 0) {
            swap(nums[left], nums[i]);
            left++;
        }
    }
}

这个写法更 "紧凑",但本质和 "填非零 + 补零" 是一个逻辑 ------ 只是把 "赋值" 换成了 "交换"~

相关推荐
二哈赛车手6 小时前
新人笔记---ApiFox的一些常见使用出错
java·笔记·spring
为何创造硅基生物7 小时前
C语言 结构体内存对齐规则(通俗易懂版)
c语言·开发语言
吃好睡好便好7 小时前
在Matlab中绘制横直方图
开发语言·学习·算法·matlab
栗子~~7 小时前
JAVA - 二层缓存设计(本地缓冲+redis缓冲+广播所有本地缓冲失效) demo
java·redis·缓存
星寂樱易李7 小时前
iperf3 + Python-- 网络带宽、网速、网络稳定性
开发语言·网络·python
YDS8297 小时前
DeepSeek RAG&MCP + Agent智能体项目 —— RAG知识库的搭建和接口实现
java·ai·springboot·agent·rag·deepseek
仰泳之鹅7 小时前
【C语言】自定义数据类型2——联合体与枚举
c语言·开发语言·算法
之歆7 小时前
DAY_12JavaScript DOM 完全指南(二):实战与性能篇
开发语言·前端·javascript·ecmascript
jolimark8 小时前
C语言自学攻略:小白入门三步走
c语言·编程入门·学习路线·实践项目·自学攻略
未若君雅裁8 小时前
MyBatis 一级缓存、二级缓存与清理机制
java·缓存·mybatis