LC 26. 删除有序数组中的重复项

题目描述

这是 LeetCode 上的 26. 删除有序数组中的重复项 ,难度为 简单

给你一个有序数组 nums ,请你原地删除重复出现的元素,使每个元素只出现一次,返回删除后数组的新长度。

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

说明:

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

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

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

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

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

示例 1:

1
2
3
4
5
输入:nums = [1,1,2]

输出:2, nums = [1,2]

解释:函数应该返回新的长度 2 ,并且原数组 nums 的前两个元素被修改为 1, 2 。不需要考虑数组中超出新长度后面的元素。

示例 2:
1
2
3
4
5
输入:nums = [0,0,1,1,1,2,2,3,3,4]

输出:5, nums = [0,1,2,3,4]

解释:函数应该返回新的长度 5 , 并且原数组 nums 的前五个元素被修改为 0, 1, 2, 3, 4 。不需要考虑数组中超出新长度后面的元素。

提示:

  • $0 <= nums.length <= 3 \times 10^4$
  • $-10^4 <= nums[i] <= 10^4$
  • nums 已按升序排列

双指针

一个指针 i 进行数组遍历,另外一个指针 j 指向有效数组的最后一个位置。

只有当 i 所指向的值和 j 不一致(不重复),才将 i 的值添加到 j 的下一位置。

Java 代码:

1
2
3
4
5
6
7
8
9
class Solution {
public int removeDuplicates(int[] nums) {
int n = nums.length, j = 0;
for (int i = 0; i < n; i++) {
if (nums[i] != nums[j]) nums[++j] = nums[i];
}
return j + 1;
}
}

C++ 代码:
1
2
3
4
5
6
7
8
9
10
class Solution {
public:
int removeDuplicates(vector<int>& nums) {
int n = nums.size(), j = 0;
for (int i = 0; i < n; i++) {
if (nums[i] != nums[j]) nums[++j] = nums[i];
}
return j + 1;
}
};

Python 代码:
1
2
3
4
5
6
7
8
class Solution:
def removeDuplicates(self, nums: List[int]) -> int:
n, j = len(nums), 0
for i in range(n):
if nums[i] != nums[j]:
j += 1
nums[j] = nums[i]
return j + 1

TypeScript 代码:
1
2
3
4
5
6
7
function removeDuplicates(nums: number[]): number {
let n: number = nums.length, j: number = 0;
for (let i: number = 0; i < n; i++) {
if (nums[i] !== nums[j]) nums[++j] = nums[i];
}
return j + 1;
};

  • 时间复杂度:$O(n)$
  • 空间复杂度:$O(1)$

通用解法

为了让解法更具有一般性,我们将原问题的「最多保留 1 位」修改为「最多保留 k 位」。

对于此类问题,我们应该进行如下考虑:

  • 由于是保留 k 个相同数字,对于前 k 个数字,我们可以直接保留。
  • 对于后面的任意数字,能够保留的前提是:与当前写入的位置前面的第 k 个元素进行比较,不相同则保留。

举个🌰,我们令 k=1,假设有样例:[3,3,3,3,4,4,4,5,5,5]

  1. 设定变量 idx,指向待插入位置。idx 初始值为 0,目标数组为 []

  2. 首先我们先让第 1 位直接保留(性质 1)。idx 变为 1,目标数组为 [3]

  3. 继续往后遍历,能够保留的前提是与 idx 的前面 1 位元素不同(性质 2),因此我们会跳过剩余的 3,将第一个 4 追加进去。idx 变为 2,目标数组为 [3,4]

  4. 继续这个过程,跳过剩余的 4,将第一个 5 追加进去。idx 变为 3,目标数组为 [3,4,5]

  5. 当整个数组被扫描完,最终我们得到了目标数组 [3,4,5] 和 答案 idx3

Java 代码:

1
2
3
4
5
6
7
8
9
10
11
12
class Solution {
public int removeDuplicates(int[] nums) {
return process(nums, 1);
}
int process(int[] nums, int k) {
int idx = 0;
for (int x : nums) {
if (idx < k || nums[idx - k] != x) nums[idx++] = x;
}
return idx;
}
}

C++ 代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
class Solution {
public:
int removeDuplicates(vector<int>& nums) {
return process(nums, 1);
}
int process(vector<int>& nums, int k) {
int idx = 0;
for (int x : nums) {
if (idx < k || nums[idx - k] != x) nums[idx++] = x;
}
return idx;
}
};

Python 代码:
1
2
3
4
5
6
7
8
9
10
11
class Solution:
def removeDuplicates(self, nums):
return self.process(nums, 1)

def process(self, nums, k):
idx = 0
for x in nums:
if idx < k or nums[idx - k] != x:
nums[idx] = x
idx += 1
return idx

TypeScript 代码:
1
2
3
4
5
6
7
8
9
10
11
function process(nums: number[], k: number): number {
let idx: number = 0;
nums.forEach((x) => {
if (idx < k || nums[idx - k] !== x) nums[idx++] = x;
});
return idx;
}

function removeDuplicates(nums: number[]): number {
return process(nums, 1)
};

  • 时间复杂度:$O(n)$
  • 空间复杂度:$O(1)$

基于上述解法我们还能做一点小剪枝:利用目标数组的最后一个元素必然与原数组的最后一个元素相同进行剪枝,从而确保当数组有超过 k 个最大值时,数组不会被完整扫描。

但需要注意这种「剪枝」同时会让我们单次循环的常数变大,所以仅作为简单拓展。

Java 代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
class Solution {
public int removeDuplicates(int[] nums) {
int n = nums.length;
if (n <= 1) return n;
return process(nums, 1, nums[n - 1]);
}
int process(int[] nums, int k, int max) {
int idx = 0;
for (int x : nums) {
if (idx < k || nums[idx - k] != x) nums[idx++] = x;
if (idx - k >= 0 && nums[idx - k] == max) break;
}
return idx;
}
}

C++ 代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
class Solution {
public:
int removeDuplicates(vector<int>& nums) {
int n = nums.size();
if (n <= 1) return n;
return process(nums, 1, nums[n - 1]);
}
int process(vector<int>& nums, int k, int max) {
int idx = 0;
for (int x : nums) {
if (idx < k || nums[idx - k] != x) nums[idx++] = x;
if (idx - k >= 0 && nums[idx - k] == max) break;
}
return idx;
}
};

Python 代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
class Solution:
def removeDuplicates(self, nums):
n = len(nums)
if n <= 1: return n
return self.process(nums, 1, nums[-1])

def process(self, nums, k, max_val):
idx = 0
for x in nums:
if idx < k or nums[idx - k] != x:
nums[idx] = x
idx += 1
if idx - k >= 0 and nums[idx - k] == max_val:
break
return idx

TypeScript 代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
function process(nums: number[], k: number, max: number): number {
let idx: number = 0;
for (const x of nums) {
if (idx < k || nums[idx - k] !== x) nums[idx++] = x;
if (idx - k >= 0 && nums[idx - k] === max) break;
}
return idx;
}
function removeDuplicates(nums: number[]): number {
const n: number = nums.length;
if (n <= 1) return n;
return process(nums, 1, nums[n - 1]);
};

  • 时间复杂度:$O(n)$
  • 空间复杂度:$O(1)$

最后

这是我们「刷穿 LeetCode」系列文章的第 No.26 篇,系列开始于 2021/01/01,截止于起始日 LeetCode 上共有 1916 道题目,部分是有锁题,我们将先把所有不带锁的题目刷完。

在这个系列文章里面,除了讲解解题思路以外,还会尽可能给出最为简洁的代码。如果涉及通解还会相应的代码模板。

为了方便各位同学能够电脑上进行调试和提交代码,我建立了相关的仓库:https://github.com/SharingSource/LogicStack-LeetCode

在仓库地址里,你可以看到系列文章的题解链接、系列文章的相应代码、LeetCode 原题链接和其他优选题解。