原地移除数组重复元素问题总结
注:本文章中的总结和源代码均来自labuladong的算法秘籍,写下此文的目的仅是为了自己复习所用。版权归原作者所有。如有侵权,请联系删除。
类型一:有序数组/链表去重
思路:快慢指针
首先声明两个指针:
int slow = 0 , fast = 0;
让fast指针一直在前面移动,当碰到不等于slow值的时候,进入if语句,
让slow++,然后nums[ slow ] = nums[ fast ]
上述思路执行完之后,slow指针就指向了当前去重后的数组的最后一个元素。
所以如果要返回元素的个数,就return slow + 1;
对于链表,基本思路完全一样,只是变成链表操作。
类型二:移除元素
思路:
代码思路和类型一是完全相同的,只不过处理方法有些不同。
在本题中,没说这个数组是有序的,因此在处理的时候也不需要考虑这个。
遇到不等于val的,也就是需要留下的,就放在前面,
nums[slow] = nums[fast];
slow++;
即可。
最大的区别就是这两行代码的顺序,并且最后返回的值也直接是slow
class Solution {
public int removeElement(int[] nums, int val) {
if(nums.length == 0) return 0;
int slow = 0 , fast = 0;
// Arrays.sort(nums);
while(fast < nums.length) {
if(nums[fast] != val) { // 把不等于val的数值筛选出来,放在最前面
// 注意 这里是先赋值 再把slow++ 和之前的数组去重不太一样
nums[slow] = nums[fast];
slow++;
}
fast++;
}
return slow;
}
}
类型三:移动元素
思路:
类型二是删除val值的元素,这里是移动val值元素。
区别就是,对于类型二,后面的元素可以不管了
但是对于类型三,必须要把后面的元素重新赋值。
如果是移动0元素,就直接把后面的全部赋值0即可。
先调用移除元素的方法,得到slow值,即length
然后用while循环赋值即可
class Solution {
public void moveZeroes(int[] nums) {
int len = removeElement(nums,0); // len是0之外的元素的个数
while(len < nums.length) {
nums[len] =0;
len++;
}
}
// 移除val值的元素
public int removeElement(int[] nums, int val) {
if(nums.length == 0) return 0;
int slow = 0 , fast = 0;
// Arrays.sort(nums);
while(fast < nums.length) {
if(nums[fast] != val) { // 把不等于val的数值筛选出来,放在最前面
// 注意 这里是先赋值 再把slow++ 和之前的数组去重不太一样
nums[slow] = nums[fast];
slow++;
}
fast++;
}
return slow;
}
}