题目:
给定一个数组 nums,编写一个函数将所有 0 移动到数组的末尾,同时保持非零元素的相对顺序。
示例:
输入:
[0,1,0,3,12]
输出:
[1,3,12,0,0]
说明: 必须在原数组上操作,不能拷贝额外的数组。 尽量减少操作次数。
分析题意:
①将所有零移动到数组的末尾:数组中所有非零元素放在数组前面,数组中所有的0放在数组末尾。
②相对顺序:所有非零元素根据原来的排列顺序,不能改变。
思路1:
这道题目很好想的一个方法就是暴力算法:将数组从前到后遍历,如果遇到零,就把该元素后面的每个数据向前移动一位,同时在数组后面加一个零。
代码如下:
for(int i=0;i<nums.size();i++)//遍历数组中的每一个元素
{
if(nums[i]==0)//当遇到0元素时
{
for(int j=i;j<nums.size()-1;j++)//将该元素后面的每一位数据向前移动一位
{
nums[j]=nums[j+1];
}
nums.push_back(0);//在数组后加一个零
}
}
暴力解决方法是非常容易想到的一个思路。但计算机是个文艺青年,他喜欢不喜欢暴力这个词,而且喜欢方便简单的生活方式。通过分析,该算法时间复杂度O(n²),提交上去,发现超时了。
既然这样,想个别的方法罢:
思路2:
其实对数组的很多问题,我们都可以用双指针的思路的去解决。双指针是解决顺序表问题的一个很好用的方法,其大体上分为两种:左右指针和快慢指针。其中,快慢指针往往用于解决链表中的相关问题,左右指针往往用于解决数组的相关问题。
左右指针没有什么固定的用法,基本就是设置两个指向数组元素的指针,通过操作指针指向数组的位置来实现对数组元素的操作。示例如下图:
该题的双指针的思路及代码:
以[1,0,2,3,0,1,25]数据为例,该题解决思路如图:
step1:
首先定义left和right指针,都指向数组的第一个元素。
step2:
然后,我们开始进行循环,每次循环,right指针向右移动一个位置,当right指针遍历完整个数组时,循环结束。
step3:
每当right指针遇到0时,我们交换两个指针指向的元素,同时使left指针向右移动一个位置;
right指针再向右移动一位:
第一次循环结束。按此规律,进行整个循环。程序结束。
作者:力扣 (LeetCode)
链接:https://leetcode-cn.com/leetbook/read/top-interview-questions-easy/x2ba4i/
来源:力扣(LeetCode)