刷题之旅从数组类型的题目开始。第二道题目是删除排序数组中的重复项,对应leetcode的题号为26。

image

题目描述

给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度。

不要使用额外的数组空间,你必须在原地修改输入数组并在使用 O(1) 额外空间的条件下完成。

  • 示例 1:

给定数组 nums = [1,1,2],

函数应该返回新的长度 2, 并且原数组 nums 的前两个元素被修改为 1, 2。

你不需要考虑数组中超出新长度后面的元素。

  • 示例 2:

给定 nums = [0,0,1,1,1,2,2,3,3,4],

函数应该返回新的长度 5, 并且原数组 nums 的前五个元素被修改为 0, 1, 2, 3, 4。

你不需要考虑数组中超出新长度后面的元素。
说明:

为什么返回数值是整数,但输出的答案是数组呢?

请注意,输入数组是以“引用”方式传递的,这意味着在函数里修改输入数组对于调用者是可见的。

你可以想象内部操作如下:

1
2
3
4
5
6
7
8
// nums 是以“引用”方式传递的。也就是说,不对实参做任何拷贝
int len = removeDuplicates(nums);

// 在函数里修改输入数组对于调用者是可见的。
// 根据你的函数返回的长度, 它会打印出数组中该长度范围内的所有元素。
or (int i = 0; i < len; i++) {
  print(nums[i]);
}

解题思路

根据题目要求,只能原地进行排除,并且还是一个排序的数组,那么肯定需要两个指针来实现,一个指针负责掌管前面已经不重复的元素索引,后者负责遍历所有元素,只要有不重复的,那么就将j所在的元素赋予给i后面一个位置的元素,然后i和j分别后移一位即可。那么到最后,i指向的元素以及前面所有元素都是不重复的。最终j遍历到数据结束为止。

提交代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
class Solution {
public int removeDuplicates(int[] nums) {
//分析数组为空或者只有一个元素的情况,返回数组长度即可
if(nums == null || nums.length <= 0 || nums.length == 1){
return nums.length;
}
//两个指针,i指向的元素,前面都是确定不重复的,最后返回长度的时候即(i+1)就是i所有不重复元素的个数
int i=0,j=1;
while(j < nums.length){
//不相等的时候,则将j的元素赋予给i+1后面一个元素,并且各自后移一位
//比如[1,2],此时将j=1的值赋给i=1的值即可,最后数组还是[1,2]
//比如[1,1,2],第一步是进入else即j=2,下一步j=2赋予给i=1,那么数组最终为[1,2,2],返回i+1即2即可。
if(nums[i] != nums[j]){
//这一行是本题精华,j指向的元素挪到i后面一个元素,i和j分别加一
nums[++i] = nums[j++];
}else{
//相等,那么j后移一位即可
j++;
}
}
return i+1;
}
}