【每日一道算法题】数组移除元素——双指针法

目录

每日一题

1.移除元素

2.数组“删除”的核心

3.双指针法

4.我的刷题实录

4.1.提交第一版

4.2.提交第二版

4.3.提交第三版

4.4.参考答案

每日一题

1.移除元素

力扣题目链接

给你一个数组 nums 和一个值 val,你需要原地移除所有数值等于 val 的元素,并返回移除后数组的新长度。

什么是原地? 我的理解就是原地不用创建新数组,在原有数组上进行删除操作。

不要使用额外的数组空间,你必须仅使用 O(1) 额外空间并原地修改输入数组。

元素的顺序可以改变。你不需要考虑数组中超出新长度后面的元素。

示例 1: 给定 nums = [3,2,2,3], val = 3, 函数应该返回新的长度 2, 并且 nums 中的前两个元素均为 2,其余两个元素不用管是什么。 你不需要考虑数组中超出新长度后面的元素。

示例 2: 给定 nums = [0,1,2,2,3,0,4,2], val = 2, 函数应该返回新的长度 5, 并且 nums 中的前五个元素为 0, 1, 3, 0, 4,其余三个元素不用管是什么。

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

2.数组“删除”的核心

关于数组删除元素的理论核心:数组一旦创建,其元素的个数(数组的长度)就不能再发生变化。数组的元素在内存地址中是连续的,不能单独删除数组中的某个元素,只能覆盖。

3.双指针法

什么是双指针法?

双指针,指的是在遍历的过程中,不是普通的使用单个指针进行访问,而是使用两个相同方向(快慢指针)或者相反方向(对撞指针)的指针进行扫描,从而达到相应的目的。

快慢指针也是双指针,但是两个指针从同一侧开始遍历数组,两个指针以不同的策略移动,直到两个指针的值相等(或其他特殊条件)为止。通过一个快指针和慢指针在一个for循环下完成两个for循环的工作。

举个栗子:假设有一个数组[2,3],要删除的元素是2,删除之后返回新数组的长度。

快慢指针法的过程是怎样的?

快指针做的是常规的遍历操作,有两种情况:

  • 当快指针遍历到的值不等于要删除的目标值val,说明该元素可取,将快指针指向的元素赋值给慢指针指向的元素;
  • 当快指针遍历到的值等于要删除的值val,此时慢指针不动,快指针继续移动到下一个,然后重复刚才的判断,当遍历的值不等于要删除的目标值val,说明该元素可取,将快指针指向的元素赋值给慢指针指向的元素,然后快指针继续移动,重复刚才的判断。
  • 总结:无论如何快指针都会正常遍历,而慢指针是遇到了目标元素就会停止,直到快指针遇到了第一个非目标元素为止,慢指针才会恢复移动。

摘自B站评论区,一种更好理解的方式:但其实slow指针指的并不是新数组的末尾,slow-1才真正指向新数组的末尾。分析过程参考下面“我的刷题实录”。

4.我的刷题实录

4.1.提交第一版
class Solution {
public:
    int removeElement(vector<int>& nums, int val) {

        int slow = 0;
        //双指针法,快指针用于正常遍历查询,慢指针用于指向新数组的末尾元素
        for(int fast = 0; fast < nums.length; fast++){
            //当快指针遍历到的值不等于目标值val,那么执行两个操作:
            //1:慢指针将这个值纳入新数组的末尾(赋值))
            //2:指向下一个元素,继续表示新数组的末尾元素(自增); 
            if(nums[fast] != val){
                nums[slow] == nums[fast];
                slow++;
            }
            //当快指针遍历到的值等于目标值val,那么慢指针停止移动(不能纳入新数组);
        }

    }
};

显示编译出错,原因是把=赋值不小心写成了==。而且缺少return语句(忘记了题目的要求是返回新数组的长度)。

4.2.提交第二版

class Solution {
    public int removeElement(int[] nums, int val) {
        int slow = 0;
        //双指针法,快指针用于正常遍历查询,慢指针用于指向新数组的末尾元素
        for(int fast = 0; fast < nums.length; fast++){
            //当快指针遍历到的值不等于目标值val,那么执行两个操作:
            //1:慢指针将这个值纳入新数组的末尾(赋值))
            //2:指向下一个元素,继续表示新数组的末尾元素(自增); 
            if(nums[fast] != val){
                nums[slow] = nums[fast];
                slow++;
            }
            //当快指针遍历到的值等于目标值val,那么慢指针停止移动(不能纳入新数组);
        }
        //在快指针遍历完成后,直接返回slow的值。sslow表示的是新数组末尾的元素,因此新数组长度为slow+1
        return slow+1;
    }
}

显示解答出错,测试用例显示:

分析了一下原因,关键在于slow的含义没搞清楚。slow实际上并不指向新数组的末尾,想象一下如果快指针此时遍历到了数组的最后一个元素,而且这个元素不是要删除的值,那么就会照常执行赋值、slow自增操作,注意在slow自增前的索引,也就是slow-1这个索引,才指向的是新数组的末尾,而slow一定指向的是新数组末尾后面一个元素    。

还是用上面的图。假设有一个数组[2,3],要删除的元素是2,删除之后返回新数组的长度。

根据图示来看,slow实际上是在赋值之后,才真正指向新数组的末尾索引,在这个过程之后还会自增,自增后实际上新数组末尾的索引应该是slow-1。而新数组的长度就是slow。

4.3.提交第三版
class Solution {
    public int removeElement(int[] nums, int val) {
        int slow = 0;
        //双指针法,快指针用于正常遍历查询,慢指针用于指向新数组的末尾元素
        for(int fast = 0; fast < nums.length; fast++){
            //当快指针遍历到的值不等于目标值val,那么执行两个操作:
            //1:慢指针将这个值纳入新数组的末尾(赋值)
            //2:指向下一个元素,继续表示新数组的末尾元素(自增); 
            if(nums[fast] != val){
                nums[slow] = nums[fast];
                slow++;
            }
            //当快指针遍历到的值等于目标值val,那么慢指针停止移动(不能纳入新数组);
        }
        //在快指针遍历完成后,直接返回slow的值。
        return slow;
    }
}
4.4.参考答案
class Solution {
    public int removeElement(int[] nums, int val) {
        // 快慢指针
        int slowIndex = 0;
        for (int fastIndex = 0; fastIndex < nums.length; fastIndex++) {
            if (nums[fastIndex] != val) {
                nums[slowIndex] = nums[fastIndex];
                slowIndex++;
            }
        }
        return slowIndex;
    }
}

参考的学习资料: 代码随想录

  • 30
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值