【双指针问题】LeetCode344、345、 844、283问题详解及代码实现

news2025/1/16 8:20:22

Halo,这里是Ppeua。平时主要更新C语言,C++,数据结构算法......感兴趣就关注我吧!你定不会失望。

 

🌈个人主页:主页链接

🌈算法专栏:专栏链接

     我会一直往里填充内容哒!

🌈LeetCode专栏:专栏链接         

    目前在刷初级算法的LeetBook 。若每日一题当中有力所能及的题目,也会当天做完发出

🌈代码仓库:Gitee链接

🌈点击关注=收获更多优质内容🌈

考试周终于结束啦,又要回归到日常来。

今天先把考试周做过的简单题都更新一下,从明天开始每日一更!

题目顺序由易到难,方便自己找回做题的感觉,也方便阅读.

目录

344. 反转字符串:

白话讲解:

题解:

代码实现:

LeetCode 345. 反转字符串中的元音字母:

白话讲解:

题解:

注:

代码实现:

LeetCode 844. 比较含退格的字符串:

白话讲解:

题解:

代码实现:

 优化

优化后的版本:

LeetCode 283. 移动零:

白话讲解:

这题的两种解法非常妙!都是我无法想到的  

题解1:

代码实现:

题解2:

代码实现:


344. 反转字符串:

编写一个函数,其作用是将输入的字符串反转过来。输入字符串以字符数组 s 的形式给出。

不要给另外的数组分配额外的空间,你必须原地修改输入数组、使用 O(1) 的额外空间解决这一问题。

输入:s = ["h","e","l","l","o"]
输出:["o","l","l","e","h"]
输入:s = ["H","a","n","n","a","h"]
输出:["h","a","n","n","a","H"]

白话讲解:

倒序输出整个数组

题解:

这题非常的简单,就是利用双指针的思想,从头尾开始遍历,并交换两个指针所指向的字符.虽然很简单,但也可以帮我们复习一下双指针的本质,毕竟算法题万变不离其宗.

可以直接使用swap,但Leecode上的速度评测好像没有这样手动交换的快.

代码实现:

class Solution {
public:
    void reverseString(vector<char>& s) {
        int left=0,right=s.size()-1;
        while(left<right)
            {
                char tmp=s[left];
                s[left++]=s[right];
                s[right--]=tmp;

            }
    }
};

 

LeetCode 345. 反转字符串中的元音字母:

给你一个字符串 s ,仅反转字符串中的所有元音字母,并返回结果字符串。

元音字母包括 'a''e''i''o''u',且可能以大小写两种形式出现不止一次。

输入:s = "hello"
输出:"holle"
输入:s = "leetcode"
输出:"leotcede"

白话讲解:

大意就是将题目中所有出现的元音字母(包含大小写)反转.

题解:

利用hash的思想(为什么我又不直接用hash呢?因为还没学www,下次一定)将所有需要用到的结果保存起来,也就是"AEIOUaeiou",这里我们把他放到string类中,因为它支持find函数,比较方便.

之后利用上一题的基础模板(双指针头尾遍历)

先遍历起始位置left,利用find比对当前位置是否是元音字母,不是就左移,是的话就停下,等待交换

下一步从尾开始向前遍历right,重复上面的步骤.

当指针停下时就说明left,right指向的字母都为元音,此时将他们交换即可.

注:

find函数没找到返回的是-1 

因为前两步都为自增操作,所以交换要判定一下是否越界

代码实现:

class Solution {
public:
    string reverseVowels(string s) {
        int right=s.size()-1,left=0;
        string tag="aeiouAEIOU";
        while(left<=right)
        {
            
            while(left<=right&&tag.find(s[left])==-1)left++;
            while(left<=right&&tag.find(s[right])==-1)right--;
            if(left<=right)swap(s[left],s[right]);  
            left++;
            right--;
        }
        return s;
    }

};  

 

LeetCode 844. 比较含退格的字符串:

给定 s 和 t 两个字符串,当它们分别被输入到空白的文本编辑器后,如果两者相等,返回 true 。# 代表退格字符。

注意:如果对空文本输入退格字符,文本继续为空。

输入:s = "ab#c", t = "ad#c"
输出:true
解释:s 和 t 都会变成 "ac"。
输入:s = "ab##", t = "c#d#"
输出:true
解释:s 和 t 都会变成 ""。
输入:s = "a#c", t = "b"
输出:false
解释:s 会变成 "c",但 t 仍然是 "b"。

白话讲解:

给两段字符串,#代表backspace,判断处理完后两段字符串是否相等

题解:

分析题目可知,仍然是一个双指针的问题.但我们可以用容器里的push,与pop简化这一操作.

遍历数组,若不为#则将字符存入,若为#且容器不为空则将上一个字符删除,继续遍历下一个.

最后利用string的特性直接比较并返回就可以了.

代码实现:

class Solution {
public:
    bool backspaceCompare(string s, string t) {
        string s1,t1;
        for(int i=0;i<s.size();i++)
        {
            if(s[i]=='#')
            {
                if(!s1.empty())
                    s1.pop_back();
                else continue;
            }
            else
                s1.push_back(s[i]);
        }
        for(int j=0;j<t.size();j++)
        {   
            if(t[j]=='#')
            {
                if(!t1.empty())
                    t1.pop_back();
                else continue;
            }
            else 
                t1.push_back(t[j]);
        }
        return t1==s1;
    }
};

 

 优化

这段代码有一点小小的问题,逻辑处理有点冗杂.

可以将中间判断的逻辑优化一下.

这里是先判定他是否为#,若为#则在进入新的一层判断是否为空,若不为空才删除,若为空则继续.

那我们可以抽出里面的核心逻辑

我只要知道这段是不是#,如果不是#我直接push,没有push的话说明是#,则判断他是否为空,有点像 

if(true)
    return ;
a++; 

当你true了底下就不会运行,当你false时,也不需要再去判定

优化后的版本:

class Solution {
public:
    bool backspaceCompare(string s, string t) {
        string s1,t1;
        for(int i=0;i<s.size();i++)
        {
            if(s[i]!='#')s1.push_back(s[i]);
            else if(!s1.empty())
                s1.pop_back();
        }
        for(int j=0;j<t.size();j++)
        {   
            if(t[j]!='#')t1.push_back(t[j]);
            else if(!t1.empty())
                t1.pop_back();
        }
        return t1==s1;
    }
};

 

 

LeetCode 283. 移动零:

给定一个数组 nums,编写一个函数将所有 0 移动到数组的末尾,同时保持非零元素的相对顺序。

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

输入: nums = [0,1,0,3,12]
输出: [1,3,12,0,0]
输入: nums = [0]
输出: [0]

白话讲解:

将非0元素整体往前移(不改变顺序),然后0放末尾

这题的两种解法非常妙!都是我无法想到的  

题解1:

我们要做的是将0的位置与非0 的对调,参考第一题,若采用前后指针的做法,就会破坏元素本来的顺序.所以不可行.

那我们想想临位指针的做法,参照快速排序的思想,将非0的都放到左边,其余放到右边.

所以依次遍历,先遍历右指针,当找到非0(交换因子)就停下

这时候左指针去在(0-right的范围里去找0),找到停下后交换

 

代码实现:

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

 

 

题解2:

还是两个临位指针,这个类似归并排序的思想,right指针向右进行遍历,遇到非0数就停下,将该值赋给left指针指向的位置,之后left++,right继续重复上面的步骤.

当right指针遍历到末尾时,此时说明left前的数据即为这个数组中出现的所有非0元素,此时将[left,num.size())的区间全部填充为0即可.

 

代码实现:

class Solution {
public:
    void moveZeroes(vector<int>& nums) {
        int i=0,j=0;
        while(j<nums.size())
        {
            while(j<nums.size()&&nums[j]==0)j++;
            if(nums.size()<=j)break;
            nums[i++]=nums[j++];
        }
        for(;i<nums.size();i++)nums[i]=0;

    }
};

 

 

 

完结撒花:

🌈本篇博客的内容【LeetCode344、345、 844、283问题详解及代码实现】已经结束。

🌈若对你有些许帮助,可以点赞、关注、评论支持下博主,你的支持将是我前进路上最大的动力。

🌈若以上内容有任何问题,欢迎在评论区指出。若对以上内容有任何不解,都可私信评论询问。

🌈诸君,山顶见!

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

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

相关文章

曼恩斯特在创业板注册生效:拟募资约5亿元,彭建林夫妇为实控人

2月21日&#xff0c;深圳证券交易所披露的信息显示&#xff0c;深圳市曼恩斯特科技股份有限公司&#xff08;下称“曼恩斯特”&#xff09;的注册生效。据贝多财经了解&#xff0c;曼恩斯特于2021年6月30日在创业板递交招股书&#xff0c;2022年6月15日获得上市委会议通过&…

老戏骨李立群真敢说,互联网吐槽郝蕾演技太差

说起老戏骨李立群&#xff0c;可能很多人都想不起来&#xff0c;他究竟出演过什么影视作品&#xff0c;不过这依然不能阻挡他的走红。李立群的走红非常偶然&#xff0c;因为在大陆拍戏多年&#xff0c;他已经在上海买房定居&#xff0c;当然偶尔也会去台北省亲。 在上海疫情爆发…

【C++】3.类和对象(中)

1.类的6个默认成员函数 一个类什么都没有不是空类 我们没有写相关函数 但编译器会自动生成6个默认函数 2.构造函数 1概念 构造函数是一个特殊的成员函数&#xff0c;名字与类名相同,创建类类型对象时由编译器自动调用&#xff0c;保证每个数据成员都有一个合适的初始值&…

微服务架构中的多级缓存设计还有人不懂?

今天我们来聊聊缓存这个话题&#xff0c;看看在微服务环境下如何设计有效的多级缓存架构。主要涉及三方面内容&#xff1a; Web 应用的客户端缓存&#xff1b;应用层静态资源缓存&#xff1b;服务层多级缓存。 首先&#xff0c;咱们先讲解微服务架构的多级缓存设计。 微服务…

ElasticSearch 学习笔记总结(一)

文章目录一、 数据的 分类二、 ElasticSearch 介绍三、 ElasticSearch 搭建四、正排索引 和 倒排索引五、ES HTTP 索引 操作六、ES HTTP 文档 操作七、ES HTTP 查询数据1. 条件查询2. 分页查询3. 排序查询4. 多条件查询5. 全文检索 完全匹配 高亮显示6. 聚合查询八、 ES HTTP 映…

2.22JVM

一.学习目标1)JVM内存区域划分2)JVM的类加载机制3)JVM的垃圾回收1.JVM执行流程程序在执行之前先要把Java代码转换为字节码(.class),JVM首先需要通过一定的方式类加载器把文件加载到运行时数据区,而字节码文件是JVM的一套指令集规范,并不能直接交给底层操作系统去执行,因此需要特…

【11-JVM面试专题-说说你知道的垃圾回收算法?垃圾回收器你知道吗?CMS、G1和ZGC垃圾回收器你有过了解吗?】

JVM面试专题-说说你知道的垃圾回收算法&#xff1f;垃圾回收器你知道吗&#xff1f;CMS、G1和ZGC垃圾回收器你有过了解吗&#xff1f; JVM面试专题-说说你知道的垃圾回收算法&#xff1f;垃圾回收器你知道吗&#xff1f;CMS、G1和ZGC垃圾回收器你有过了解吗&#xff1f;你掌握的…

Wincc Flexible smart V4触摸屏软件中批量导入PLC变量的具体方法示例(无需单个添加)

Wincc Flexible smart V4触摸屏软件中批量导入PLC变量的具体方法示例(无需单个添加) 具体操作步骤可参考以下例子中的内容: 打开STEP7-MicroWin smart软件,编辑自己的PLC程序(这里以一个简单的启保停程序为例), 如下图所示,打开Wincc Flexible smart V4触摸屏软件,新建…

双指针 (C/C++)

1. 双指针 双指针算法的核心思想&#xff1a;将暴力解法的时间复杂度&#xff0c;通常是O(N*N)&#xff0c;通过某种特殊的性质优化到O(N)。 做题思路&#xff1a;先想想暴力解法的思路&#xff0c;然后分析这道题的特殊性质&#xff0c;一般是单调性。然后得出双指针算法的思路…

微信协议网页版微信协议解析

最近在做个微信机器人&#xff0c;所以研究了网页版的微信协议及相关接口&#xff0c;在这里简单总结一下。从表面上看&#xff0c;对于网页版微信我们的使用流程是这样的&#xff1a;很简单&#xff0c;只有四步&#xff0c;但如果细化到内里细节的话&#xff0c;上面这简单四…

一文带你快速入门zabbix6.0的日常操作

文章目录前言一. zabbix基本操作入门1.1 登录和配置用户1.1.1 登录zabbix1.1.2 防爆力破解机制1.1.3 创建用户1.1.4 创建报警媒介1.1.5 设置 权限选项卡1.1.6 设置用户的访问主机权限1.2 新建主机1.2.1 添加主机1.2.2 关于添加的信息注释1.3 新增监控项1.3.1 添加监控项1.3.2 配…

若依系统如何集成qq邮件发送【超详细,建议收藏】

若依系统的部署博主就不在这儿阐述了&#xff0c;默认大家的电脑已经部署好了若依系统&#xff0c;这里直接开始集成邮件系统&#xff0c;首先我们得需要对qq邮箱进行配置&#xff1b;一套学不会你来打我&#x1f600;&#xff1b; 一、开启我们的qq邮箱发送邮件的配置 1、先进…

Qt音视频开发16-通用悬浮按钮工具栏的设计

一、前言 通用悬浮按钮工具栏这个功能经过了好几个版本的迭代&#xff0c;一开始设计的时候是写在视频控件widget窗体中&#xff0c;当时功能简单就放一排按钮在顶部悬浮widget中就好&#xff0c;随着用户需求的变化&#xff0c;用户需要自定义悬浮条的要求越发强烈&#xff0…

K_A12_031 基于STM32等单片机驱动TEMT6000环境光传感器 串口与OLED0.96双显示

K_A12_031 基于STM32等单片机驱动TEMT6000环境光传感器 串口与OLED0.96双显示一、资源说明二、基本参数参数引脚说明三、驱动说明IIC地址/采集通道选择/时序对应程序:四、部分代码说明1、接线引脚定义1.1、STC89C52RCTEMT6000环境光传感器模块1.2、STM32F103C8T6TEMT6000环境光…

ZCMU--5009: 龙虎斗

轩轩和开开正在玩一款叫《龙虎斗》的游戏&#xff0c;游戏的棋盘是一条线段&#xff0c;线段上有n个兵营(自左至右编号1~n)&#xff0c;相邻编号的兵营之间相隔1厘米&#xff0c;即棋盘为长度为n-1厘米的线段。i号兵营里有ci位工兵。 下面图1为n 6的示例: 轩轩在左侧&#xf…

如何通过IP找到地址?

在我们印象中&#xff0c;我们都知道可以通过 IP 地址找到某个人。但当我们细想一下&#xff0c;我们会发现其实 IP 地址与地理位置并不是直接相关的。那我们到底是如何通过 IP 地址找到地址的呢&#xff1f;答案是&#xff1a;通过自治系统&#xff08;Autonomous System&…

大势前瞻!文旅还是短视频,你弯道超车风口在这了

三年前&#xff0c;新冠疫情的影响波及整个各行各业行业&#xff0c;互联网寒冬&#xff0c;房地产崩盘&#xff0c;教培团灭&#xff0c;在这样的背景下&#xff0c;行业都进入了发展“冰雪期”。老话说大疫后必有大变&#xff0c;如今风雪融化&#xff0c;万物复苏&#xff0…

day01_HTML常识

基础概念铺垫 认识网页 网页的组成&#xff1a;文字、图片、音频、视频、超链接一系列元素网页的本质&#xff1a;程序员写的代码通过浏览器转化&#xff08;解析和渲染&#xff09;成用户看到的网页网页是由网页元素组成的 &#xff0c; 这些元素是利用html标签描述出来&…

面试攻略,Java 基础面试 100 问(十八)

JAVA IO 包 JAVA NIO NIO 主要有三大核心部分&#xff1a;Channel(通道)&#xff0c;Buffer(缓冲区), Selector。 传统 IO 基于字节流和字 符流进行操作&#xff0c;而 NIO 基于 Channel 和 Buffer(缓冲区)进行操作&#xff0c;数据总是从通道读取到缓冲区 中&#xff0c;或者…

不可替代的测试人:一文解释探索性测试是什么

剧本式测试 随着软件技术的发展&#xff0c;软件开发过程中对软件测试的需求越来越庞大&#xff0c;从原来单纯的寻找软件缺陷&#xff0c;到后来的学习软件、挖掘软件中存在的缺陷、评估软件可用性、性能等等方面&#xff0c;软件测试发挥着重要作用。 考虑到软件测试如此的…