Problem: 283. 移动零
文章目录
- 思路
- 算法图解分析
- 复杂度
- Code
思路
首先我们来讲一下本题的思路
- 本题主要可以归到【数组划分/数组分块】这一类的题型。我们将一个数组中的所有元素划分为两段区间,左侧是非零元素,右侧是零元素
- 那解决这一类的题我们首先想到的就是【双指针算法】,学习过C语言的同学应该就可以知道指针是比较繁琐和复杂,如果有兴趣学习的同学可以看看我的这篇文章 链接
- 不过在这里呢我们不需要去使用
int*
这种指针,而是直接使用数组下标来充当指针即可
好,那我们就来看看这个双指针到底是怎样的,要如何去使用
- 两个指针的作用
- 【cur】: 从左往右扫描数组,遍历数组
- 【dest】:已处理的区间内,非零元素的最后一个位置
- 可以看到,
cur
是我们用来遍历数组的,从[cur, n - 1]
就是还未处理的元素;那么从[0, cur]
就是已经处理过的元素,但是呢本题的要求是我们要划分出【零元素】与【非零元素】,所以呢前面的区间我们可以再度划分为[0, dest]
和[dest + 1, cur - 1]
小结一下:
[0, dest] [dest + 1, cur - 1] [cur, n - 1]
[0, dest]
—— 非零元素[dest + 1, cur - 1]
—— 零元素[cur, n - 1]
—— 未处理元素
算法图解分析
接下去我们就通过画算法图解的形式来模拟一下解题的过程
- 我们就以题目中所给出的第一个示例为例来进行讲解,因为在一开始我们还没处理过任何的非零元素,所以对于
[0, cur - 1]
这段区间是没有任何数据的,所以在一开始我们可以将【dest】这个指针置于-1的位置
- 因为我们需要将非0元素移动到前面,所以呢如果遇到了0元素的话,
cur++
即可,将其留在这个位置上
- 那当我们遇到非0元素时,就需要将其交换到前面去,那我们
[0, dest]
这个区间就是用来存放非0元素的,此时多了一个元素的话那dest
就要加1,原本其是指向-1这个位置,那我们可以使用++dest
来完成
- 接下去,当数据交换过来后,我们可以去对照上面的这三个区间,可以发现最左侧是非0元素,中间是0元素,右侧呢则是待处理的元素。接下去我们又碰到了0元素,所以
cur++
cur
再后移之后呢,我们又碰到了非0元素,继续让dest
上来然后交换二者位置上的元素
- 那现在我们再来看这三个区间,左侧还是保持为【非0元素】,中间为【0元素】,右侧的话则是【待处理的元素】
- 然后碰到非0元素后,继续让
++dest
,然后做交换
- 最后的话我们来看看这个处理完后的整个区间元素:非0元素都在前面,而0元素则都在后面,
[cur, n - 1]
的这段区间也不存在了,说明已经没有待处理元素了
复杂度
接下去我们来分析一下本题的时空复杂度
- 时间复杂度:
本算法的核心思路参考的是【快速排序】的区间划分,我们这里就是在不断遍历数组的过程中,以中间的0作为分割,然后左侧是非0元素,右侧是未处理的元素。在处理的过程中我们只是遍历了一次这个数组,所以复杂度为 O ( n ) O(n) O(n)
- 空间复杂度:
在本题中我们并没有去开出额外的空间,所以复杂度为 O ( 1 ) O(1) O(1)
Code
class Solution {
public:
void moveZeroes(vector<int>& nums) {
for(int dest = -1, cur = 0; cur < nums.size(); ++cur)
{
if(nums[cur] != 0)
{
swap(nums[++dest], nums[cur]);
}
}
}
};