双指针算法——部分OJ题详解

news2025/1/11 15:07:45

目录

关于双指针算法:

1,对撞指针

2,快慢指针

部分OJ题详解 

283.移动零

1089.复写零

202.快乐数

11.盛水最多的容器

611.有效三角形的个数

剑指offer 57.和为s的两个数字

15.三数之和

18.四数之和


关于双指针算法:

双指针算法是一种经典算法,一般有两种:对撞指针和快慢指针 。

1,对撞指针

一般用于顺序结构中,也叫做左右指针。意思是从定长顺序结构的最左端和最右端向中间移动,当在循环内部找到结果或者满足:left == rightleft > right时就退出循环

2,快慢指针

和如名字一样,两个指针在顺序结构的同一端同时移动,一个指针一次往后移动一次,一个一次移动两次。该算法非常适合用于处理数组和环形数据结构问题。

部分OJ题详解 

283.移动零

283. 移动零 - 力扣(LeetCode)

 该题目可归类为“数组划分,数组分块”问题,可以使用快慢指针将整个数组分成0区间和非0区间。

定义两个指针:cur和dest。其中cur的作用很简单,就是从左往右遍历数组,dest的作用则是划分0区间和非0区间,表示已经划分好的0区间的最后一个如下图:

class Solution {
public:
    void moveZeroes(vector<int>& nums) {
        for(size_t cur = 0, dest = -1; cur < nums.size(); cur++)
        {
            if(nums[cur] != 0) 
            {
                swap(nums[++dest], nums[cur]);
            }
        }
    }
};

1089.复写零

1089. 复写零 - 力扣(LeetCode)

该题目简单解释就是:给一个数组arr,将每一个出现的0往后复写一位,不覆盖后面的数据,其余元素往后移动,不要越界,只能修改原数组。

咱们来分析这道题:

  1. 首先我们要找到最后一个要复写的数的位置下标,定义双指针cur=0,dest=-1,cur先开始遍历,如果cur不为0,dest和cur都走一步,如果cur为0,dest走两步,最后当dest走到数组结尾时,开始复写0
  2. 有个特例,[1, 0, 2, 3, 0 ,4],该数组按照上面的步骤,最后的cur会指向第二个0,dest最后走到了第7个位置也就是数组后面一个位置,但是由于cur为0,所以在复写的时候会把dest改为0,造成越界访问,所以需要处理下越界情况
  3. 当最后dest == arr.size()时,arr[n - 1] = 0; dest += 2;
class Solution {
public:
    void duplicateZeros(vector<int>& arr)
    {
        //1,先找到最后一个数
        int cur = 0, dest = -1;
        int n = arr.size();
        while(cur < n)
        {
            if(arr[cur] == 0)
            {
                dest += 2;
            }
            else
            {
                dest ++;
            }

            if(dest >= n - 1) break;
            //当dest合法时,在cur++
            cur++;
        }
        //2,处理边界情况
        if(dest == n)
        {
            arr[n - 1] = 0;
            cur--;
            dest += 2;
        }
        //3,开始从后往前复写0
        while(cur >= 0)
        {
            if(arr[cur] != 0)
            {
                arr[dest--] = arr[cur--]; 
            }
            else
            {
                arr[dest--] = 0;
                arr[dest--] = 0; //复写0
                cur--;
            }
        }
    }
};

202.快乐数

202. 快乐数 - 力扣(LeetCode)

 题目有点复杂,但是相信大家看来示例之后很快能理解,下面我们来分析下这个题目:

  1. 这道题一共有三种情况,第一种情况是:和示例一样,输入19,最后会变成1,变成1之后会陷入无限循环    第二种情况是:输入2,无限循环,但始终变不到1    第三种情况就是始终变不到1并且无重复数字,但该题不考虑
  2. 以第一第二种情况为例,最后都会变成一个数并且无限循环,也就是说两种情况带入链表的话最后会有一个“环”,所以这道题和我们的“判断链表是否有环”很相似,这道题我们只需要判断环种的数是否为1即可
  3. 所以我们采用快慢指针来解,然后因为的特性,只要判断两个指针相遇时的值即可
  4. 最后,我们可以直接用数来代替指针的租用,假设slow指针=2,那么平方之后变成了4,就相当于在数组中对指针进行了++,这是慢指针,所以快指针进行两次平方,也就是++两次
class Solution {
public:
    int bitSum(int n) //返回n数每一位上的平方和
    {
        int sum = 0;
        while(n > 0)
        {
            int t = n % 10;
            sum += t*t;
            n /= 10;
        }
        return sum;
    }
    bool isHappy(int n) 
    {
        int slow = n, fast = n;
        fast = bitSum(fast);
        while(fast != slow)
        {
            slow = bitSum(slow);
            fast = bitSum(fast);
            fast = bitSum(fast);
        }
        return slow == 1;
    }
};

11.盛水最多的容器

11. 盛最多水的容器 - 力扣(LeetCode)

解释一下题目:给一个长度为n的整数数组height,每个元素代表一个垂线表示容器的高度,找出其中两条线,使得它们与x轴共同构成的容器可以容纳最多的水,返回容器可以存储的最大水量。下面我们来分析下这道题:

  1. 首先是暴力解法:用两个for循环将所有的可容纳的水的容量全部枚举出来,列成数组然后排序,选出最大值。这种解法肯定会超时,因为时间复杂度为O(N*2)
  2.  我们先选一个区间:[6, 2, 5, 4],首先计算6到4的容量,为6 * 3 = 18,当向内枚举时,也就是2和4,5和4,容器的宽度是一直在减小的,但是高度会有两种情况:当向内枚举是碰见了一个小的数(2比4小),所以高度减小;当碰到了一个大的数(5比4大),虽然高度不变,但是宽度减小了,所以总体容量还是在减小。那么我们就可以去掉2和4,5和4的枚举,就能减少枚举次数
  3. 然后就是总体了[1, 8, 6, 2, 5, 4, 8, 3, 7]。先算1和7,得出v1,1比7小,所以左指针++;计算8到7,得出v2,8比7大,所以右指针--;再算8到3,得出v3,8比3大,所以右指针++
  4. 像这样利用单调性和双指针来搞,只需要遍历一遍数组就可以了,就可以把暴力解法的O(N * 2)变成了O(N)
class Solution {
public:
    int maxArea(vector<int>& height) 
    {
        int left = 0, right = height.size()-1, ret = 0;
        while(left < right) //阻止条件就是两个指针都往中间移动,直到相遇就停下来
        {
            int v = min(height[left], height[right]) * (right - left);
            ret = max(ret, v);
            //计算完毕,移动指针,谁小移动谁
            if(height[left] < height[right])
            {
                left++;
            }
            else
            {
                right--;
            }
        }
        return ret;
    }
};

611.有效三角形的个数

611. 有效三角形的个数 - 力扣(LeetCode)

解释下这道题:给我们一个数组,选择数组中任意三个数尝试组成一个三角形,返回能够组成三角形的个数,下面我们来分析下这道题:

  1. 首先是暴力解法,既然要找三个数,那就用三个循环穷举所有情况,然后只需判断a+b是否大于c即可,但是我们可以先对数组排序,这样原本要三个数各判断一次,排序后只需判断一次
  2. 排完序后再进行列举,只有两种情况:①a+b>c    ②a+b <= c,我们只需对这两个情况做判断即可
  3. 假设要处理的数组为[2, 2, 3, 4, 5, 9, 10],我们可以先固定最大值10,然后定义双指针,left指向2,right指向9;2+9>10 符合情况①,这时候可以看到2后面的值都比2大,那么2后面的数和9相加肯定是大于10的,所以就不要再left++了,直接将后面的情况全部纳入,也就是可形成的三角形数直接+(right-left),就可以省去很多次计算和比较
  4. 那么left不要动,那么自然是right往左移了,right接着指向5,这时候2+5<10,符合情况②,那么right就不要再往左走了,因为5左边的数肯定比5小,不可能组成三角形,所以要left++再判断
  5. 然后按照上面的步骤把固定10的情况全搞定,然后再固定9,这样依次判断,就能用一个循环解决问题
class Solution {
public:
    int triangleNumber(vector<int>& nums) 
    {
        //1,排序优化
        sort(nums.begin(), nums.end());

        //2,采用双指针解决问题
        int ret = 0;//统计最终个数
        int n = nums.size();
        for(int i = n-1; i >= 2; i--)
        {
            //利用双指针快速统计符合要求的三元组个数
            int left = 0, right = i - 1;
            while(left < right)
            {
                //判断能否构成三角形
                if(nums[left] + nums[right] > nums[i]) //符合情况①
                {
                    ret += right - left;
                    right--;
                }
                else //符合情况②
                {
                    left++;
                }
            }
        }
        return ret;
    }
};

剑指offer 57.和为s的两个数字

LCR 179. 查找总价格为目标值的两个商品 - 力扣(LeetCode)

解释下这道题:给一个升序的数组和一个数字s,在数组里找两个数使和等于s,如果有多种结果返回一组即可,下面我们来分析下这道题:

  1. 首先还是暴力解法:两个for循环穷举全部情况,一个一个相加判断即可,但是暴力解法没有用到数组已经有序这个条件,所以这道题我们还是用单调性和双指针来解决。
  2. 假设给的数组是[2, 7, 11, 15, 19, 21],s=30;所以当相加的时候会有三种情况①sum>t    ②sum<t    ③sum=t。
  3. 首先定义left指针指向2,right指针指向21,2+21<30,符合情况②,所以21前面的数都小于21,所以直接全部排除,right不动,left++到7的时候仍然小于20,left再++
  4. 当left指向11时,11+19==30,符合要求于是返回,如果再次符合情况②,left再++,如果符合情况①,那么right--
  5. 就这样我们只需要遍历一次数组就能找到等于30的两个数了
class Solution 
{
public:
    vector<int> twoSum(vector<int>& nums, int s) {
        int left = 0, right = nums.size() - 1;
        while(left < right)
        {
            if(nums[left] + nums[right] > s)
            {
                right--;
            }
            else if(nums[left] + nums[right] < s)
            {
                left++;
            }
            else
            {
                return {nums[left], nums[right]};
            }
        }
        return {-1};
    }
};

15.三数之和

15. 三数之和 - 力扣(LeetCode)

 解释一下题目:给你一个数组,在这个数组中随便选三个数使它们的和为0,并且三个数互不相同,返回所有相加为0的三个数组合,顺序没有要求但是元素的下标不能一样,假设输入:[-1, 0, 1, 2, -1, -4]  ,我们可以选出三个[-1, 0, 1] [-1, 2, -1] [0, 1, -1]但是只返回前两个,是因为第三个和第一个顺序不一样,但是元素一样的,这样的情况我们二选一返回。下面我们来分析下这道题:

  1. 首先是暴力解法,三个for暴力穷举,然后将每个选出来的三个数组成vector,然后排序+去重,去重可以用set容器去重,但是暴力解法时间复杂度为O(N^3)
  2. 其实这道题和上面的“和为s的两个数”的题目很像,只是s变成了0,两数之和变为了三数之和,所以我们也可以尝试用排序+双指针来解决
  3. 首先对原数组进行排序,假设排完序后是[-4, -4, -1, 0, 0, 0, 1, 1, 4, 4, 5, 6],可以先固定第一个-4,i=-4,然后定义双指针left指向第二个-4,right指向6,然后可以和“有效三角形个数”题目一样,找到left和right的和等于 -(-4)即可,然后再次固定-1,重复操作即可
  4. 但是这道题有一些细节需要额外处理:①不借助set去重,原数组排序后已经有序,所以找到一种结果之后,left和right都要跳过重复元素,并且使用完一个固定数i之后,i也要跳过重复元素,并且,跳过重复元素的时候也要注意越界问题,比如[0, 0, 0, 0]
  5. ②找到一种结果之后,指针不要停,要继续往里缩继续找
class Solution {
public:
    vector<vector<int>> threeSum(vector<int>& nums)
     {
         vector<vector<int>> vv;
        //1,排序
        sort(nums.begin(), nums.end());

        //2,利用双指针解决问题
        int n = nums.size();
        for(int i = 0; i < n;) //固定数a
        {
            if(nums[i] > 0) break; //优化:如果最小的数已经大于0,那么后面的数相加一定不等于0
            int left = i + 1, right = n - 1, a = -nums[i];
            while(left < right)
            {
                int sum = nums[left] + nums[right];
                if(sum > a)
                    right--;
                else if(sum < a)
                    left++;
                else
                {
                    vv.push_back({nums[i], nums[left], nums[right]});
                    left++;
                    right--; //不漏
                    while(left < right && nums[left] == nums[left - 1]) left++; //跳过重复元素
                    while(left < right && nums[right] == nums[right + 1] ) right--; //对指针去重
                }
            }
            //对i去重
            i++;
            while(i < n && nums[i] == nums[i - 1]) i++;
        }
        return vv;
    }
};

18.四数之和

18. 四数之和 - 力扣(LeetCode)

解释下题目:和三数之和题目一样,只是变成了4个数的和为0,下面来分析下这道题:

  1. 暴力解法就是还是4次循环 + set去重,但是时间复杂度为O(N^4),巨耗时间
  2. 其实我们可以复用我们三数之和的算法,固定一个数,然后在这个数后面使用三数之和算法,然后再把固定的数往后面挪,再使用三数之和,依次循环就可以解决四数之和的问题
class Solution {
public:
    vector<vector<int>> fourSum(vector<int>& nums, int target) 
    {
         vector<vector<int>> vv;
        //1,排序
        sort(nums.begin(), nums.end());

        //2,利用双指针解决问题
        int n = nums.size();
        for(int a = 0; a < n;) //固定数a
        {
            //利用三数之和解决问题
            for(int b = a + 1; b < n;)
            {
                //双指针
                int left = b + 1, right = n -1;
                long long aim = (long long)target - nums[a] - nums[b]; //利用双指针找到两个数的和等于aim即可
                while(left < right)
                {
                    int sum = nums[left] + nums[right];
                    if(sum < aim) 
                        left++;
                    else if(sum > aim)
                        right--;
                    else
                    {
                        vv.push_back({nums[a], nums[b], nums[left++], nums[right--]}); //找到一种结果
                        //去重操作 --> ①先去重left和right,②再去重b,③最后再a去重
                        //去重①
                        while(left < right && nums[left] == nums[left-1]) left++;
                        while(left < right && nums[right] == nums[right+1]) right--;
                    }
                }
                //去重②
                b++;
                while(b < n && nums[b] == nums[b-1]) b++;
            }
            a++;
            while(a < n && nums[a] == nums[a-1]) a++;
        }
        return vv;
    }
};

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1854585.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

硬盘数据恢复软件,推荐5种适合你的方法来恢复硬盘数据

硬盘数据恢复软件&#xff0c;作为解决数据丢失问题的关键工具&#xff0c;帮助用户在重要文件丢失时迅速找回数据。本教程介绍5种恢复实用硬盘数据方法&#xff0c;适应不同类型和严重程度的数据损坏情况。 文章摘要&#xff1a; 一. 硬盘数据恢复软件 二. 数据恢复原理 三. …

ThinkPHP:查询数据库数据之后,更改查询数据的字段名称

一、原始查询数据 含有字段item_no&#xff0c;lot_num&#xff0c;position $data[brushed] db::table(wip_station_transaction) ->where([wip_entity_name>$wip_entity_name,line_code>$line_code,]) ->field([item_no, lot_num, position]) ->select(); …

React18中各种Hooks用法总结( 内附案例讲解)

React中各种Hooks用法总结 内附案例讲解 一、useState useState 是一个 React Hook&#xff0c;它允许你向组件添加一个 状态变量。 import React, { FC, memo, useState } from react import { MainContainer } from ./style interface IProps {children?: React.ReactNo…

上新:NFTScan 正式上线 Bitcoin-brc20 浏览器!

近日&#xff0c;NFTScan 团队正式对外发布了 Bitcoin-brc20 浏览器&#xff0c;将为 Bitcoin 生态的 NFT 开发者和用户提供简洁高效的 NFT 数据搜索查询服务。作为比特币生态中最火热的标准之一&#xff0c;brc20 也吸引着广泛的关注。洞悉其巨大潜力&#xff0c;NFTScan 对 b…

基于springboot websocket和okhttp实现消息中转

1、业务介绍 消息源服务的消息不能直接推给用户侧&#xff0c;用户与中间服务建立websocket连接&#xff0c;中间服务再与源服务建立websocket连接&#xff0c;源服务的消息推给中间服务&#xff0c;中间服务再将消息推送给用户。流程如下图&#xff1a; 此例中我们定义中间服…

Linux应急响应——知攻善防应急靶场-Linux(1)

文章目录 查看history历史指令查看开机自启动项异常连接和端口异常进程定时任务异常服务日志分析账户排查总结 靶场出处是知攻善防 Linux应急响应靶机 1 前景需要&#xff1a; 小王急匆匆地找到小张&#xff0c;小王说"李哥&#xff0c;我dev服务器被黑了",快救救我&…

【React】ref

概述 使用 ref 引用值 – React 中文文档 希望组件“记住”某些信息&#xff0c;但又不想让这些信息更新时 触发新的渲染 时&#xff0c;可以使用 ref 。 也就是说 ref 对象 包裹的值 React 追踪不到的&#xff0c;他像是用来存储组件信息的秘密“口袋”。 与 state 相同的是…

一、系统学习微服务遇到的问题集合

1、启动了nacos服务&#xff0c;没有在注册列表 应该是版本问题 Alibaba-nacos版本 nacos-文档 Spring Cloud Alibaba-中文 Spring-Cloud-Alibaba-英文 Spring-Cloud-Gateway 写的很好的一篇文章 在Spring initial上面配置 start.aliyun.com 重新下载 < 2、 No Feign…

美团携手HarmonyOS SDK,开启便捷生活新篇章

华为开发者大会&#xff08;HDC 2024&#xff09;于6月21日在东莞松山湖拉开序幕&#xff0c;通过一系列精彩纷呈的主题演讲、峰会、专题论坛和互动体验&#xff0c;为开发者们带来了一场知识与技术的盛宴。6月23日&#xff0c;《HarmonyOS开放能力&#xff0c;使能应用原生易用…

上位机图像处理和嵌入式模块部署(mcu和swd接口)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 最近学习mcu的时候&#xff0c;接触了不少调试器&#xff0c;这里面有daplink、st-link v2、j-link v9。虽然模块的形状可能不太一样&#xff0c;但…

程序人生:关于RHCE红帽认证这件事

花了两个月备考红帽&#xff0c;最终终于双满分通过。 关于考试 RHCE红帽认证总共需要考两门&#xff1a;RHCSA、RHCE。 RHCSA主要是考察基本的Linux操作&#xff1a;用户、权限、空间扩容、yum、容器等内容。 RHCE主要是考察ansible playbook 代码的开发。 通过考试没有别…

【web1】标签,css,js

文章目录 1.标签&#xff1a;input1.1 html&#xff1a;HTML&#xff08;用于创建网页结构&#xff09;&#xff0c;CSS&#xff08;对页面进行美化&#xff09;&#xff0c;JavaScript&#xff08;用于与用户交互&#xff09;1.2 文本标签&#xff1a;字体属性1.3 a标签&#…

DIVE INTO DEEP LEARNING 50-55

文章目录 50. semantic segmentation50.1 Basic concepts50.2 Major application 51. Transposed convolution51.1 Basic concepts51.2 Major role51.3 Implementation steps and application areas51.4 Transposed convolution51.5 Transposed convolution is a type of convo…

卧槽,6。套死你猴子,Tomcat访问html页面显示源码?

卧槽&#xff0c;6。Tomcat访问html页面显示源码&#xff1f; 元凶text/explain //踩坑&#xff01;&#xff01;&#xff01;不能用 servletResponse.setContentType("text/explain&#xff0c;否则访问html会看到源码&#xff0c;而不是渲染页面; charsetUTF-8"…

IDEA各种实体类运行爆红,不运行就没事

1.问题描述 如图所示&#xff0c;后端项目的import的各种entity爆红&#xff0c;点击也有导入包的提示&#xff0c;且这种报红几乎遍布了整个工程项目 2.我的解决方案 清空缓存&#xff0c;然后把target文件删掉&#xff0c;重新跑 3.小结 idea项目有时候就是一个核弹&…

【科普】半导体制造过程的步骤、技术、流程

在这篇文章中&#xff0c;我们将学习基本的半导体制造过程。为了将晶圆转化为半导体芯片&#xff0c;它需要经历一系列复杂的制造过程&#xff0c;包括氧化、光刻、刻蚀、沉积、离子注入、金属布线、电气检测和封装等。 基本的半导体制造过程 1.晶圆&#xff08;Wafer&#xf…

免费一年SSL证书申请——建议收藏

免费一年SSL证书申请——建议收藏 获取免费一年期SSL证书其实挺简单的 准备你的网站&#xff1a; 确保你的网站已经有了域名&#xff0c;而且这个域名已经指向你的服务器。还要检查你的服务器支持HTTPS&#xff0c;也就是443端口要打开&#xff0c;这是HTTPS默认用的。 验证域…

23.并发

目录 一、一些概念二、进程和线程2.1 概念2.2 多线程导致的问题2.3 使用spawn创建新线程2.4 线程与move闭包 三、消息传递3.1 概念3.2 创建通道3.3 示例3.4 其它测试 四、共享状态并发4.1 互斥器4.2 Mutex的API4.3 多线程共享Mutex1&#xff09;在多线程之间共享值&#xff1b;…

ARC学习(3)基本编程模型认识(三)

笔者来介绍arc的编程模型的中断流程和异常流程 1、中断介绍 主要介绍一下中断进入的流程&#xff0c;包括需要配置的寄存器等信息。 中断号&#xff1a;16-255&#xff0c;总共240个中断。触发类型&#xff1a;脉冲或者电平触发中断优先级&#xff1a;16个&#xff0c;0最大&…

在linux系统中使用docker、mysql实例

systemctl 是一个命令行工具&#xff0c;用于控制和管理基于 systemd 的 Linux 发行版中的系统和服务。 启动服务 &#xff1a;使用 systemctl start [service-name] 开始一个服务。 如启动docker&#xff1a;systemctl start docker 停止服务 &#xff1a;使用 systemctl st…