【代码随想录刷题记录】LeetCode844比较含退格的字符

题目地址

1. 思路

1.1 基本思路

拿到这个题,我们要单独写一个函数去将退格后的字符串结果返回出来(生成退格后的真实的字符串),我还是想魔改 O ( n ) O(n) O(n)时间复杂度的删除数组元素的算法:【代码随想录刷题记录】LeetCode27移除元素

那我们就需要思考一下,这个算法中的slow和fast指针的关系,我们都知道,当遇到要删除的元素,slow会停下来,而fast会继续自增,而我们要删除的元素的范围实际上是[slow, fast),左闭右开,我们看一下之前的代码:

cpp 复制代码
class Solution {
public:
    // 引用传递,直接改nums,是改其本身,不是拷贝
    int removeElement(vector<int>& nums, int val) {
        int slow = 0; //慢指针,用来记录删除该元素后,每个元素对应的新下标slow
        for(int fast = 0; fast < nums.size(); fast++)
        {
            if(val != nums[fast])
            {
                // 下一次循环必须先执行赋值操作再进行slow的自增
                nums[slow] = nums[fast];
                slow++;
            }       
        }
        return slow;
    }
};

假设我们要删除的是3,数组nums是[1,1,3,3,2]
fast=0

由于 nums[fast] = nums [ 0 ] = 1 ≠ val = 3 \text{nums[fast]}=\text{nums}[0]=1\ne\text{val}=3 nums[fast]=nums[0]=1=val=3,则

fast=1

由于 nums[fast] = nums [ 1 ] = 1 ≠ val = 3 \text{nums[fast]}=\text{nums}[1]=1\ne\text{val}=3 nums[fast]=nums[1]=1=val=3,则

fast=2

由于 nums[fast] = nums [ 2 ] = 3 = val = 3 \text{nums[fast]}=\text{nums}[2]=3=\text{val}=3 nums[fast]=nums[2]=3=val=3,则什么也不做,进入下一次循环

fast=3

由于 nums[fast] = nums [ 3 ] = 3 = val = 3 \text{nums[fast]}=\text{nums}[3]=3=\text{val}=3 nums[fast]=nums[3]=3=val=3,则什么也不做,进入下一次循环
fast=4

由于 nums[fast] = nums [ 4 ] = 2 ≠ val = 3 \text{nums[fast]}=\text{nums}[4]=2\ne\text{val}=3 nums[fast]=nums[4]=2=val=3,则

这个时候,循环就停止了,slow是新数组的长度,但是我们仔细看到这个删除的过程,在最后一轮循环的时候,删除的范围是[slow, fast),即

那我们再一次回看这个删除的区间和现在的这个题目,我们要删除的是#和#之前的字符,那我们的删除的区间就是[slow-1,fast),那就说明, fast所指元素在遍历的时候如果是#(也就是增加了一个else条件),那就应该让slow自减,但是slow自减只有在slow下标合法也即slow不是0的时候才能自减,我们把上面那个数组换成字符串即[a,b,#,#,c]

如果我们想要单纯删除#,当然还可以保持原有的逻辑,但是我们要删除的区间要扩大,相当于退了两个单位(假设没有nums[slow]=nums[fast]这个过程)

这个才是我们理想的删除情况,它的具体过程应该追溯到fast为2的时候(因为之前的流程和没加else条件的slow--是一样的):
fast=2

由于 nums[fast] = nums [ 2 ] = # = val = # \text{nums[fast]}=\text{nums}[2]=\#=\text{val}=\# nums[fast]=nums[2]=#=val=#,则slow--

fast=3

由于 nums[fast] = nums [ 3 ] = # = val = # \text{nums[fast]}=\text{nums}[3]=\#=\text{val}=\# nums[fast]=nums[3]=#=val=#,则slow--

此时,我们成功找到了删除的区间
fast=4

由于 nums[fast] = nums [ 4 ] = c ≠ val = # \text{nums[fast]}=\text{nums}[4]=\text{c}\ne\text{val}=\# nums[fast]=nums[4]=c=val=#,则

这个时候,slow对应的是新字符串的长度,新字符串中只有一个字符c

但是单纯地加上这样一个else条件也是不行的,如果字符串是[#, #, c]的话,最开始遇到#退格,slow会向左越界,会变成-1,所以只有在slow没到0的时候(也即大于0的时候)才能slow自减。
我们生成退格后的真实的字符串后,再对比两个字符串就能返回出结果

1.2 代码实现

从上面的内容得知,代码实现如下,这次没有做验证,直接顺利通过:

cpp 复制代码
class Solution {
public:
    //生成退格后的真实的字符串
    string genRealString(string s)
    {
        int n = s.size(); // 字符串长度
        int slow = 0;
        string result = ""; // 要返回的字符串
        for (int fast = 0; fast < n; fast++)
        {
            //当前遇到不是#就自增,和O(n)时间复杂度删除数组中某个元素的方法类似
            if (s[fast] != '#')
            {
                s[slow] = s[fast];
                slow++;
            }
            //不同于删除数组中某个元素的方法,它需要在遇到#后让slow后退一个位置
            //因为我们要删除的是#之前的字符
            else
            {
                //如果slow没向左越界就自减,因为假设#在下标为0的位置,则其退格
                //相当于没退格,但是#需要删除,所以此时slow不需要自减1
                //其他情况需要自减1(将要删除元素的范围是#和其前面的元素,
                //删除元素的范围是[slow,fast),所以slow自减是为了将#之前的元素删除
                if(slow > 0)
                {
                    slow--;
                }
            }
        }
        result = s.substr(0, slow); // 截取从0开始,长度为slow的字符串
        return result;
    }
    bool backspaceCompare(string s, string t) {
        string a = genRealString(s);
        string b = genRealString(t);
        return a == b;
    }
};
相关推荐
int型码农4 小时前
数据结构第八章(一) 插入排序
c语言·数据结构·算法·排序算法·希尔排序
UFIT4 小时前
NoSQL之redis哨兵
java·前端·算法
喜欢吃燃面4 小时前
C++刷题:日期模拟(1)
c++·学习·算法
SHERlocked934 小时前
CPP 从 0 到 1 完成一个支持 future/promise 的 Windows 异步串口通信库
c++·算法·promise
怀旧,4 小时前
【数据结构】6. 时间与空间复杂度
java·数据结构·算法
积极向上的向日葵5 小时前
有效的括号题解
数据结构·算法·
GIS小天5 小时前
AI+预测3D新模型百十个定位预测+胆码预测+去和尾2025年6月7日第101弹
人工智能·算法·机器学习·彩票
_Itachi__5 小时前
LeetCode 热题 100 74. 搜索二维矩阵
算法·leetcode·矩阵
不忘不弃5 小时前
计算矩阵A和B的乘积
线性代数·算法·矩阵
不爱写代码的玉子5 小时前
HALCON透视矩阵
人工智能·深度学习·线性代数·算法·计算机视觉·矩阵·c#