前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >【优选算法】探索双指针之美(一):初识双指针

【优选算法】探索双指针之美(一):初识双指针

作者头像
_孙同学
发布2024-10-21 21:06:45
850
发布2024-10-21 21:06:45
举报
文章被收录于专栏:技术分享

前言:

双指针顾名思义就是用两个指针相互配合来解决问题的。这两个指针可以在同一个数组或链表上,也可以在不同的数据结构上。它们既可以同时移动,也可以一快一慢。

作用: 使用双指针可以提高效率,在一次遍历中就可以解决问题,避免了重复遍历和不必要的计算。

1.1 移动零

题目链接: 283.移动零 题目叙述: 给定一个数组 nums,编写一个函数将所有 0 移动到数组的末尾,同时保持非零元素的相对顺序。

请注意 ,必须在不复制数组的情况下原地对数组进行操作。

解题思路: 双指针算法(利用数组下标充当指针)

1. 定义两个指针

cur:从左向右扫描数组,遍历数组。 dest:已经处理的区间内,非零元素的最后一个位置。

2. 两个指针将区间分为3部分

[0,dest]:非零元素 [dest+1,cur-1]:零元素 [cur,n-1]:待处理元素

3. 过程模拟

   ①让cur指向下标为0的位置

   ②让dest指向-1的位置(因为dest是非零元素的最后一个位置,刚开始时不知道第一个位置是否为非零元素)

   ③ 让cur进行扫描,cur会遇到两种情况:零元素和非零元素;     当遇到0元素时,cur++,就可以满足区间[dest+1,cur-1]为零元素     当遇到非零元素时,先让dest++,然后交换curdest位置上的值,cur++进行下一个数据的扫描

4. 代码实现

代码语言:javascript
复制
class Solution {
public:
    void moveZeroes(vector<int>& nums) {
        for(int cur = 0,dest = -1;cur < nums.size();cur++)
        {
            if(nums[cur])//处理非零元素
            swap(nums[++dest],nums[cur]);
        }
    }
};

1.2 复写零

题目链接:1089.复写零 题目叙述: 给你一个长度固定的整数数组 arr ,请你将该数组中出现的每个零都复写一遍,并将其余的元素向右平移。

注意:请不要在超过该数组长度的位置写入元素。请对输入的数组 就地 进行上述修改,不要从函数返回任何东西

解题思路: 先根据“异地”操作,然后优化成双指针下的“就地”操作

1.异地操作

    ① 定义一个cur指针,指向数组中的第一个元素     ②定义一个dest指针,指向开辟数组中最终的位置     ③当cur扫描时碰到非零元素,dest位置插入这个非零元素,cur++,dest++(两个指针同时向右挪动一位)

   ④ 当cur扫描时碰到零元素,dest位置插入0,dest向右再挪动一位插入0cur++,dest++(两个指针同时向右挪动一位)    ⑤ [1,0,2,3,0,4,5,0] [1,0,0,2,3,0,0,4]

2.就地操作

当我们进行就地操作时,会发现从左向右操作时会覆盖掉数组中原来的数据,所以我们需要从右往左进行操作

从右往左进行操作时就需要首先找到cur指向的最后一个复写的数.

2.1先找到最后一个复写的数

    ①先判断cur位置的值,为0 ,dest移动2步;不为0dest移动1步     ②移动dest     ③判断dest是否为最后一个位置     ④cur向下移动一位

2.2 处理边界情况

当我们找最后一个复写的数时会发现dest指向的位置会有两种情况

      ①判断是否dest > n-1       ② 若大于将n-1指向的位置赋0       ③ cur--,dest-=2

2.3 "从后向前"完成复写

   ①保证cur > 0    ②如果cur指向的位置不为零,将cur位置的值给destcur--,dest--    ③如果cur指向的位置为零,将dest位置的值赋0dest++,再将dest位置的值赋为0;    ④cur--,dest--

3.代码实现

代码语言:javascript
复制
class Solution {
public:
    void duplicateZeros(vector<int>& arr) {
        //1.找到最后一个复写的数
        int cur = 0,dest = -1; int n = arr.size();
        while(cur < n)
        {
            if(arr[cur]) dest++;
            else dest+=2;
            if(dest >= n - 1) break;
            cur++;
        }
        //2.处理边界情况
        if(dest > n-1)
        {
            arr[n-1] = 0;
            cur--;
            dest-=2;
        }
        //"从前向后"完成复写
        while(cur>=0)
        {
            if(arr[cur])
            {
                arr[dest--] = arr[cur--];

            }
            else
            {
                arr[dest--] = 0;
                arr[dest--] = 0; 
                cur--;
            }
        }
    }
};

1.3 快乐数

题目链接:202.快乐数 题目叙述: 编写一个算法来判断一个数 n 是不是快乐数。

「快乐数」 定义为:   对于一个正整数,每一次将该数替换为它每个位置上的数字的平方和。 然后重复这个过程直到这个数变为 1,也可能是 无限循环 但始终变不到 1。

如果这个过程 结果为 1,那么这个数就是快乐数。 如果 n 是 快乐数 就返回 true ;不是,则返回 false 。

示例 1:

输入:n = 19 输出:true 解释:   12 + 92 = 82   82 + 22 = 68   62 + 82 = 100   12 + 02 + 02 = 1

示例 2:

输入:n = 2 输出:false

解题思路: 类似判断链表是否有环->抽象出的链表:判断环里的数是否为1

解法: 快慢双指针

    ①定义快慢指针     ②慢指针每次向后移动一步,快指针每次向后移动两步     ③判断相遇时候的值

解题步骤: 1.定义一个函数获取每一位的数字的平方和 2.定义快慢指针 fast 指向第二个数, slow 指向第一个数 3.让slow 走一步,fast 走两步,直到两个相遇为止 4. 判断相遇位置的值

代码实现:

代码语言:javascript
复制
class Solution
{
public:
    int Sum(int n)//返回n这个数每一位上的平方和
    {
        int sum = 0;
        while(n)
        {
            int t = n % 10;
            sum += t * t;
            n = n / 10;
        }
        return sum;
    }
    bool isHappy(int n)
    {
        int slow = n,fast = Sum(n);
        while(slow != fast)
        {
            slow = Sum(slow);
            fast = Sum(Sum(fast));
        }
        return slow == 1;
    }
};

最后

以上三道题就是对双指针算法的初步应用,可以帮助我们初步了解并熟悉双指针算法,欲知后事如何,关注我请听下回分解

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2024-10-15,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 前言:
  • 1.1 移动零
  • 1.2 复写零
  • 1.3 快乐数
  • 最后
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档