题目描述:
给你一个数组 nums 和一个值 val,你需要原地移走所有数值等于 val 的元素。元素的顺序可能发生改变。然后返回 nums 中与 val 不同的元素的数量。
注意:
假设 nums 中不等于 val 的元素数量为 k,要通过此题,您需要执行以下操作:
- 更改 nums 数组,只需要使 nums 的前 k 个元素都是不等于 val 的元素。nums 的其余元素和 nums 的大小并不重要。
- 返回 k的值。
输入输出示例:
输入:nums = [3,2,2,3], val = 3
输出:2, nums = [2,2,_,_]
解释:你的函数函数应该返回 k = 2, 并且 nums 中的前两个元素均为 2。你在返回的 k 个元素之外留下了什么并不重要(因此它们并不计入评测)。
方式一:双指针
算法思想:将不等于val的数组元素往前放,右指针 right 指向当前将要处理的元素,左指针 left 指向下一个将要赋值的位置。从前往后处理
实现代码:
class Solution {
public int removeElement(int[] nums, int val) {
int n = nums.length;
int left = 0;
//循环处理数组的每个元素,从前往后遍历
for (int right = 0; right < n; right++) {
//right指针发现值不等于val的元素,就将元素放到数组前面
if (nums[right] != val) {
nums[left] = nums[right];
left++;
}
}
return left;
}
}
特殊情况:
示例1:
输入:nums = [3,2,2,3], val = 3
输出:2, nums = [2,2,_,_]
两个指针都遍历了一遍数组
示例2:
输入:nums = [3,2,2,3], val = 2
输出:2, nums = [3,3,_,_]
复杂度分析:
时间复杂度:O(n)
其中 n 为序列的长度。我们只需要遍历该序列至多两次。
空间复杂度:O(1)。
我们只需要常数的空间保存若干变量。
方法二:双指针优化 最优解法
算法思想:将等于val的数组元素往后放,两个指针初始时分别位于数组的首尾,向中间移动遍历该序列。右指针 right 指向当前将要处理的元素,左指针 left 指向下一个将要赋值的位置。
实现代码:
class Solution {
public int removeElement(int[] nums, int val) {
int left = 0;
int right = nums.length;
while (left < right) {
if (nums[left] == val) {
nums[left] = nums[right - 1];
right--;
} else {
left++;
}
}
return left;
}
}
特殊情况:
示例1:
输入:nums = [3,2,2,3], val = 3
输出:2, nums = [2,2,_,_]
两个指针一共遍历了一遍数组。
示例2:
输入:nums = [3,2,2,3], val = 2
输出:2, nums = [3,3,_,_]
复杂度分析
时间复杂度:O(n)
其中 n 为序列的长度。我们只需要遍历该序列至多一次。
空间复杂度:O(1)。
我们只需要常数的空间保存若干变量。
欢迎大家点赞,收藏,加关注呦!!!