前言

哈喽,我是长路,目前刚刚大三,方向是后端也偶尔捣鼓下前端,现在的主语言是Java。之前一大段时间都是在学习web开发的一些技术,就很久没有进行类似于数据结构、算法之类的学习与刷题,打算这段时间拾起来好好学一学、搞一搞。

这段时间也是机缘巧合看到草帽路飞的博客,加了自学群,正巧看到博主组织在群里组织了leetcode刷题打卡活动,我也就参与进来,为期一个月,打算坚持每天都花一些时间做一些题目,并通过博客的方式来进行记录。

目前跟着一个Github仓库刷题(leetcode):代码随想录leetcode刷题,当前为栈与队列专题。



题目

题目来源leetcode

leetcode地址:1047. 删除字符串中的所有相邻重复项,难度:简单。

题目描述(摘自leetcode):

给出由小写字母组成的字符串 S,重复项删除操作会选择两个相邻且相同的字母,并删除它们。

在 S 上反复执行重复项删除操作,直到无法继续删除。
在完成所有重复项删除操作后返回最终的字符串。答案保证唯一。

示例:
输入:"abbaca"
输出:"ca"
解释:
例如,在 "abbaca" 中,我们可以删除 "bb" 由于两字母相邻且相同,这是此时唯一可以执行删除操作的重复项。之后我们得到字符串 "aaca",其中又只有 "aa" 可以执行重复项删除操作,所以最后的字符串为 "ca"。
 
提示:
1 <= S.length <= 20000
S 仅由小写英文字母组成。

本地调试代码:

class Solution {
    public String removeDuplicates(String s) {
        ...
    }

    public static void main(String[] args) {
        System.out.println(new Solution().removeDuplicates("abbaca"));
    }
}


题解

NO1:栈解决

思路:遍历字符时,每次先判断当前栈顶元素是否与要入栈的元素相同,如果相同出栈,不相同入栈。最后将栈中的字符进行一一拼接返回。

示例:"abbaca" stack=[]
① 字符'a',当前栈为空直接入栈  stack=['a']
② 字符'b',栈不为空,与栈顶元素a比较,不相同入栈  stack=['a','b']
③ 字符'b',栈不为空,与栈顶元素b比较,相同出栈  stack=['a']
④ 字符'a',栈不为空,与栈顶元素b比较,相同出栈  stack=[]
⑤ 字符'c',当前栈为空直接入栈  stack=['c']
⑥ 字符'a',栈不为空,与栈顶元素c比较,不相同入栈  stack=['a','c']
最终从前往往后将元素移出进行拼接,返回"ac"

代码

class Solution {
    public String removeDuplicates(String s) {
        Deque<Character> stack = new LinkedList<>();
        for (char c : s.toCharArray()) {
            //当前栈不为空,且栈顶与当前字符相同出栈
            if(!stack.isEmpty() && stack.peek() == c){
                stack.pop();
            }else{//否则直接入栈
                stack.push(c);
            }
        }
        StringBuilder str = new StringBuilder();
        while(!stack.isEmpty()){
            str = str.append(stack.pollLast());
        }
        return str.toString();
    }
}

leetcode【栈与队列—简单】 1047. 删除字符串中的所有相邻重复项_职场和发展



NO2:字符串作栈

思路:使用字符串作栈的好处就是可以省去上面提交中拼接的操作,最终留在字符串里的就是要返回出去的。

示例:"abbaca"  str="" top=-1
① 字符'a' 字符串()为空,直接拼接 str="a"  top=0
② 字符'b' 字符串()不为空,与栈顶不相同,直接拼接 str="ab" top=1
③ 字符'b' 字符串()不为空,与栈顶相同,删除指定元素 str="a" top=0
④ 字符'a' 字符串()不为空,与栈顶相同,删除指定元素 str="" top=-1
⑤ 字符'c' 字符串()为空,直接拼接 str="c"  top=0
⑥ 字符'b' 字符串()不为空,与栈顶不相同,直接拼接 str="ca" top=1
最终直接返回str="ca"   

代码

class Solution {
    //目的是拿到不重复的字符拼接内容
    public String removeDuplicates(String s) {
        //直接来拿字符串来作为栈进行操作,最终剩下来的就是不重复的
        StringBuilder str = new StringBuilder();
        int top = -1;
        for (char c : s.toCharArray()) {
            if(top >= 0 && c == str.charAt(top)){
                str.deleteCharAt(top--);
            }else{
                str.append(c);
                top++;
            }
        }
        return str.toString();
    }
}

leetcode【栈与队列—简单】 1047. 删除字符串中的所有相邻重复项_算法_02



NO3、双指针(原数组上操作)

思路:直接对原数组进行操作,不相邻的重复元素直接覆盖旧元素,最终直接截取原数组指定长度内容返回。

fast指针用来进行遍历一遍字符串的,[0,slow)永远表示的是非相邻重复项
示例:s="abbaca"  slow=0,fast=0
① 字符'a' s[0]=s[0] (s="abbaca") slow=1,fast=1  | [0,slow)=> "a"
② 字符'b' s[1]=s[1] (s="abbaca") slow=2,fast=2  | [0,slow)=> "ab"
③ 字符'b' s[2]=s[2] (s="abbaca")(s[2]==s[1] => b=b重复,slow-1) slow=1,fast=3    | [0,slow)=> "a"
④ 字符'a' s[1]=s[3] (s="aabaca")(s[1]==s[0] => a=a重复,slow-1) slow=0,fast=4    | [0,slow)=> ""
⑤ 字符'c' s[0]=s[4] (s="cabaca")(slow=0,slow+1) slow=1,fast=5     | [0,slow)=> "c"
⑥ 字符'a' s[1]=s[5] (s="cabaca")(s[1]!=s[0],slow++) slow=1,fast=4    | [0,slow)=> "ca"
最终直接返回"ca"即可

代码

//快慢指针,时间复杂度O(n),空间复杂度O(1)
public String removeDuplicates(String s) {
    //定义双指针
    int slow = 0;//左指针的目的是①检测是否有相等,有后退,无前进。②实时更新当前最新位置值(方便下次进行对比以及旧值的覆盖,旧的值已无用)
    int fast = 0;//右指针负责的工作是进行遍历作用
    char[] chars = s.toCharArray();
    while(fast < s.length()){
        chars[slow] = chars[fast];
        if(slow > 0 && chars[slow]==chars[slow-1]){
            slow--;
        }else{
            slow++;
        }
        fast++;
    }
    //[0,slow)区间值
    return new String(chars,0,slow);
}

leetcode【栈与队列—简单】 1047. 删除字符串中的所有相邻重复项_算法_03



参考文章

[1]. leetcode题解

[2]. 代码随想录—删除字符串中的所有相邻重复项


我是长路,感谢你的耐心阅读。如有问题请指出,我会积极采纳!
欢迎关注我的公众号【长路Java】,分享Java学习文章及相关资料
Q群:851968786 我们可以一起探讨学习
注明:转载可,需要附带上文章链接