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

每日一题系列(day 08)

前言:

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

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


题目:

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

示例:

提示:

  • 1 <= nums.length <= 104
  • -231 <= numsi <= 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语言语法中的指针,可以是数组下标,或者某个索引等...

相关推荐
QiLinkOS5 小时前
第三视觉理解徐玉生与他的商业活动(30)
大数据·c++·人工智能·算法·开源协议
疯狂打码的少年5 小时前
【操作系统】页面置换算法(OPT/FIFO/LRU)
算法
小O的算法实验室5 小时前
2026年CIE,优化客货协同运输:综合地铁系统的列车容量动态分配
算法
Coder_Shenshen6 小时前
西门子S7CommPlus协议鉴权算法原理与流程详解
网络·后端·算法
硕风和炜7 小时前
【LeetCode: 2492. 两个城市间路径的最小分数 + DFS】
java·算法·leetcode·深度优先·dfs·bfs·并查集
我是一颗柠檬8 小时前
【Java项目技术亮点】加权轮询负载均衡算法
java·算法·负载均衡
灯厂码农8 小时前
C语言动态内存分配完全指南(malloc、calloc、realloc、free)
java·c语言·算法
凯瑟琳.奥古斯特9 小时前
K次取反最大化数组和解法(力扣1005)
开发语言·c++·算法·leetcode·职场和发展
Jerry9 小时前
LeetCode 203. 移除链表元素
算法
地平线开发者10 小时前
征程 6 | 工具链 QAT ObserverBase 源码解析
算法