每日一题:LeetCode-283. 移动零

每日一题系列(day 08)

前言:

🌈 🌈 🌈 🌈 🌈 🌈 🌈 🌈 🌈 🌈 🌈 🌈 🌈

🔎🔎如果说代码有灵魂,那么它的灵魂一定是👉👉算法👈👈,因此,想要写出💚优美的程序💚,核心算法是必不可少的,少年,你渴望力量吗😆😆,想掌握程序的灵魂吗❓❗️那么就必须踏上这样一条漫长的道路🏇🏇,我们要做的,就是斩妖除魔💥💥,打怪升级!💪💪当然切记不可😈走火入魔😈,每日打怪,拾取经验,终能成圣🙏🙏!开启我们今天的斩妖之旅吧!✈️✈️


题目:

给定一个数组 nums,编写一个函数将所有 0 移动到数组的末尾,同时保持非零元素的相对顺序。请注意 ,必须在不复制数组的情况下原地对数组进行操作。

示例:

提示:

  • 1 <= nums.length <= 104
  • -231 <= nums[i] <= 231 - 1

思路:

这题其实很简单,题目要求我们在不改变原有数组的顺序下将零元素全部移动到数组的最右侧。我把这类题目称为划分数组的题目,而解决这类将数组划分为几个区域的问题,我们经常会使用双指针算法。

首先题目给我们一个数组,让我们最终得到的结果被划分成两个区间,一个是元素为0的区间,一个是元素不为0的区间。但是这是将数组最终得到的结果,在我们没划分完的时候我们其实可以分为三个区间:

首先我们先设置一个指针cur用来遍历这个数组,一个指针dest表示已处理的非零元素的最后一个位置。在cur走完之前其实可以分为三个区间:

当cur走完之后就只剩下两个区间:

开始,dest指向-1,cur从0开始往后遍历,只要遇到不是0元素就先将dest+1(因为开始指向-1,所以每次交换前要先+1)在交换这两个值,到cur指向n的时候遍历结束,我们就得到了目标结果。

其实这个过程仔细观察,我们会发现其实这个过程和我们的快排很类似,快排是选取一个基准值将小于基准值的放左区间,大于基准值的放右区间。对于这题来说,基准值就是0。只不过大于0的在左边而已。

代码实现:

cpp 复制代码
class Solution {
public:
    void moveZeroes(vector<int>& nums) {
        int cur = 0;
        int dest = -1;//首先指向-1
        while(cur != nums.size())//cur不超过数组下标范围就循环
        {
            if(nums[cur])//不为0就交换
            {
                swap(nums[++dest], nums[cur]);//别忘记++dest
            }
            ++cur;//最后cur向后走一步
        }
    }
};

双指针问题在数组划分类似的问题中非常常见,双指针并不一定是C语言语法中的指针,可以是数组下标,或者某个索引等...

相关推荐
CoovallyAIHub16 分钟前
Pipecat:构建实时语音 AI Agent 的开源编排框架,500ms 级端到端延迟
深度学习·算法·计算机视觉
灰色小旋风20 分钟前
力扣13 罗马数字转整数
数据结构·c++·算法·leetcode
2301_8101609524 分钟前
C++与物联网开发
开发语言·c++·算法
cm65432028 分钟前
基于C++的操作系统开发
开发语言·c++·算法
ArturiaZ31 分钟前
【day57】
开发语言·c++·算法
CoovallyAIHub32 分钟前
Energies | 8版YOLO对8版Transformer实测光伏缺陷检测,RF-DETR-Small综合胜出
深度学习·算法·计算机视觉
Emberone40 分钟前
排序:万物皆有序
算法·排序算法
其实秋天的枫41 分钟前
2025年12月英语六级真题及答案解析完整版(第一、二、三套全PDF)
经验分享·算法
2401_874732531 小时前
C++并发编程中的死锁避免
开发语言·c++·算法
2301_792308251 小时前
C++编译期数学计算
开发语言·c++·算法