【优选算法】Pointer-Slice:双指针的算法切片(上)

news/2024/12/24 6:55:30 标签: 算法, c++, java, 数据结构, leetcode

文章目录

  • 1.概念解析
  • 2.移动零
  • 3.复写零
  • 4.快乐数
  • 5.盛最多水的容器
  • 希望读者们多多三连支持
  • 小编会继续更新
  • 你们的鼓励就是我前进的动力!

本篇是优选算法双指针算法,该算法主要用于实现特定的算法逻辑,比如查找比较排序合并等操作,降低时间复杂度,减少空间复杂度,提高程序效率🚀

1.概念解析

🚩什么是双指针算法

在这里插入图片描述

双指针算法使用两个索引来遍历数据结构,可以根据问题的要求,以不同的方式移动,如同向移动相向移动快慢不同的速度移动

2.移动零

✏️题目描述:

在这里插入图片描述

✏️示例:

在这里插入图片描述

传送门:移动零

题解:
💻第一步:

在这里插入图片描述

有两个索引 destcur
dest 表示在已处理的区间内非零元素的最后一个位置
cur 表示从左往右扫描数组遍历数组

在这里插入图片描述

把整个区间划分为三个部分,从前往后分别是非零区间0区间待处理区间

💻第二步:

根据题意我们要把 0 都移到数组末尾,所以是要注意是移动,而不是覆盖

在这里插入图片描述

刚开始dest 指向 -1的位置,表示非0区间还不存在,然后 cur 先向右移动,如果为 0 就继续向后;如果为非0,就让 dest 向后一位,然后和 cur 交换(因为 cur 遇到 0 不会改变其位置,所以在 dest 后面必定至少有一个 0,通过交换就能一直把 0 向后移)

在这里插入图片描述

总结后的代码如下:

在这里插入图片描述

💻代码实现:

#include <vector>
#include <iostream>
using namespace std;

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]);
            }
        }
    }
};

3.复写零

✏️题目描述:

在这里插入图片描述

✏️示例:

在这里插入图片描述

传送门:复写零

题解:

双指针问题在解题通常要求就地操作,但往往很难立马想出来,所以可以先进行异地操作拓展思路。本题的异地操作就是额外创建一个数组,然后据题意操作即可,很简单就不过多讲解

💻第一步: 先找到最后一个复写的数

从前往后复写会发现会出现前一个数把后一个数覆盖的情况,所以我们尝试从后往前复写,发现是可行的,所以唯一的要点就是找到那个开始复写的数

请添加图片描述

如图为示例 1找到最后一个复写的数,那么是如何找到的呢?没有过多的技巧,就是要通过不断地画图尝试找到规律

cur 表示最后一个复写的数
dest 表示是否为最后一个数

在这里插入图片描述

如果 cur 的值为 0dest 向后两位如果 cur 的值为非 0dest 向后一位。那么就延伸出另一个问题,要是 dest 越界了怎么办?

💻第二步: 处理越界情况并从后往前复写

如果越界了,那么 dest 所在的位置一般默认为 0 ,但是在平台上越界就会报错,且这种情况的时候一定是因为 cur 最后一个复写数为 0 导致的

请添加图片描述

所以我们只需将 n-1 处赋为 0dest -= 2cur-- 即可回到最后一个复写数为非0的情况

在这里插入图片描述

接着再完成从后向前复写的操作即可

💻代码实现:

#include <iostream>
#include <vector>
using namespace std;

class Solution 
{
public:
    void duplicateZeros(vector<int>& arr)
    {
        //1.先找到最后一个数
        int cur = 0, dest = -1, n = arr.size();
        while (cur < n)
        {
            if (arr[cur])
            {
                dest++;
            }
            else
            {
                dest += 2;
            }

            if (dest >= n - 1)
            {
                break;
            }

            cur++;
        }
        //2.处理边界情况
        if (dest == n)
        {
            arr[n - 1] = 0;
            dest -= 2;
            cur--;
        }
        //3.从后向前完成复写操作
        while (cur >= 0)
        {
            if (arr[cur])
            {
                arr[dest--] = arr[cur--];
            }
            else
            {
                arr[dest--] = 0;
                arr[dest--] = 0;
                cur--;
            }
        }
    }
};

4.快乐数

✏️题目描述:

在这里插入图片描述

✏️示例:

在这里插入图片描述

传送门:快乐数

题解:
💻解读题意:

据题意快乐数的判断分为两种情况

是快乐数 ,以 1 循环(以示例 1 为例)

在这里插入图片描述

不是快乐数,自循环(以示例 2 为例)

在这里插入图片描述

看到这里显然需要我们判断是否成环,在链表部分了解过,应该使用快慢指针

在这里插入图片描述

💻细节问题:

如果题目没有说明只有两种情况,那是不是可能会出现第三种情况:线性死循环
答案是不会的,以下是一些简单的证明,不影响本题,作了解即可

在这里插入图片描述

我们假设 n 可取的最大值为 9 × 10⁹ ,那么经过快乐数操作的数为 810,接下来无论进行多少次操作都是在[1,810]里的数,那么在经过大于 810 次操作后,根据鸽巢原理,必然会有重复,也就是成环,所以第三种情况不可能存在

💻代码实现:

#include <iostream>
using namespace std;

class Solution
{
public:
    int bitSum(int n)
    {
        int sum = 0;
        while (n)
        {
            int t = n % 10;
            sum += t * t;
            n /= 10;
        }
        return sum;
    }
    bool isHappy(int n)
    {
        int slow = n, fast = bitSum(n);
        while (slow != fast)
        {
            slow = bitSum(slow);
            fast = bitSum(bitSum(fast));
        }
        return slow == 1;
    }
};

5.盛最多水的容器

✏️题目描述:

在这里插入图片描述

✏️示例:

在这里插入图片描述

传送门:盛最多水的容器

题解:

看到这道题一般最先想到的是用两层for循环暴力枚举,但在本题会超时,时间复杂度为 O(n²),所以本题的思路是尽量把时间复杂度降为O(n)

尝试减少枚举数量来降低时间复杂度,本题求的是体积,所以我们可以在标记开头和结尾的下标为 left 和 right

在这里插入图片描述

v 为体积h 为高度w 为宽度,可以发现在取两边的数计算宽度时,先固定一个不动,然后另一个逐渐缩小宽度,小的那个数缩小之后算出来的体积永远是小的,所以我们可以通过不断舍掉小的那个数缩小宽度,然后得出多个体积数找出里面最大的那个

在这里插入图片描述

通过这种方式减少了不必要的枚举,降低了时间复杂度,只需遍历一遍数组就能得出最大的体积

💻代码实现:

#include <iostream>
#include <vector>
using namespace std;

class Solution 
{
public:
    int maxArea(vector<int>& height) 
    {
        int left = 0, right = height.size() - 1, ret = 0;
        while (left < right)
        {
            int v = (right - left) * min(height[left], height[right]);
            ret = max(v, ret);
            if (height[left] > height[right])
            {
                right--;
            }
            else
            {
                left++;
            }
        }
        return ret;
    }
};

希望读者们多多三连支持

小编会继续更新

你们的鼓励就是我前进的动力!

请添加图片描述


http://www.niftyadmin.cn/n/5797455.html

相关文章

SmartX分享:SMTX ZBS 中 RDMA 技术简介

目录 背景如何实现存储网络是什么TCP/IPRDMARDMA 工作原理RDMA 的实现方案 ZBS 支持 RDMA 的要求 参考 背景 我们清楚&#xff0c;分布式存储将利用网络作不同设备的互联。最基础的如TCP/IP的IP SAN&#xff0c;进阶的有FC SAN、IB等等。 SmartX 支持 10G以上的TCP/IP网络作为…

【NI国产替代】基于国产FPGA+全志T3的全国产16振动+2转速(24bits)高精度终端采集板卡

16振动2转速&#xff08;24bits&#xff09;高精度终端采集板卡 采用AG16KF256国产FPGAT3国产ARM全国产化 的处理器架构&#xff0c;设计分为2块板&#xff0c;一块底板&#xff0c; 一块核心板&#xff0c;底板负责16路信号2路转速的 采集&#xff0c;信号的滤波&#xff0…

软件测试之压力测试【详解】

压力测试 压力测试是一种软件测试&#xff0c;用于验证软件应用程序的稳定性和可靠性。压力测试的目标是在极其沉重的负载条件下测量软件的健壮性和错误处理能力&#xff0c;并确保软件在危急情况下不会崩溃。它甚至可以测试超出正常工作点的测试&#xff0c;并评估软件在极端…

GCDWebServer 使用指南

GCDWebServer 使用指南 GCDWebServer The #1 HTTP server for iOS, macOS & tvOS (also includes web based uploader & WebDAV server) [这里是图片001] 项目地址: https://gitcode.com/gh_mirrors/gc/GCDWebServer 项目介绍 GCDWebServer 是一个现代且轻量级的基于…

Linux系统编程——理解系统内核中的信号捕获

目录 一、sigaction() 使用 信号捕捉技巧 二、可重入函数 三、volatile关键字 四、SIGCHLD信号 在信号这一篇中我们已经学习到了一种信号捕捉的调用接口&#xff1a;signal(),为了深入理解操作系统内核中的信号捕获机制&#xff0c;我们今天再来看一个接口&#xff1a;si…

Leetcode打卡:考场就坐

执行结果&#xff1a;通过 题目&#xff1a; 855 考场就坐 在考场里&#xff0c;有 n 个座位排成一行&#xff0c;编号为 0 到 n - 1。 当学生进入考场后&#xff0c;他必须坐在离最近的人最远的座位上。如果有多个这样的座位&#xff0c;他会坐在编号最小的座位上。(另外&am…

重温设计模式--单例模式

文章目录 单例模式&#xff08;Singleton Pattern&#xff09;概述单例模式的实现方式及代码示例1. 饿汉式单例&#xff08;在程序启动时就创建实例&#xff09;2. 懒汉式单例&#xff08;在第一次使用时才创建实例&#xff09; 单例模式的注意事项应用场景 C代码懒汉模式-经典…

2024-05-18 前端模块化开发——ESModule模块化

目录 1、认识 ES Module2、ES Module基本使用3、export关键字 3.1、导出方式一——直接导出3.2、导出方式二——通过as起别名3.3、导出方式三——定义的时候就直接导出 4、import关键字 4.1、导入方式一——直接导入4.2、导入方式二——通过as起别名4.3、导入方式三——可以给…