LeetCode.1047-重复删除字符串中的所有相邻重复项

这是小川的第389次更新,第419篇原创

01 看题和准备

今天介绍的是LeetCode算法题中Easy级别的第251题(顺位题号是1047)。给定一个小写字母的字符串S,重复删除两个相邻且相等的字母。

我们在S上反复删除,直到我们再也无法删除。

在完成所有此类重复删除后返回最后一个字符串。保证答案是独一无二的。

例如:

输入:"abbaca"
输出:"ca"
说明:在"abbaca"中我们可以删除"bb",因为字母相邻且相等,这是唯一可能的移动。这一举动的结果是字符串是"aaca",其中只有"aa"是可能的,所以最后的字符串是"ca"。

注意

  • 1 <= S.length <= 20000
  • S仅由英文小写字母组成。

02 第一种解法

题目的意思是依次比较S中相邻的两个字母,如果相同,就删除掉,组成一个新的字符串,继续刚才的操作,直到最后S中没有相邻的重复的字母。

思路:定义一个布尔类型变量flag,控制上一次遍历S中的字母时,是否存在相邻的重复字母。使用两层循环,外层控制是否可以继续删除,内层遍历S中相邻的字母,如果相同就删掉,这里采取截取字符串的方式来生成新的字符串,同时将flag改为true,以便继续外层循环。

public String removeDuplicates(String S) {
    boolean flag = true;
    while (flag) {
        flag = false;
        int n = S.length()-1;
        for (int i=0; i<n; i++) {
            if (S.charAt(i) == S.charAt(i+1)) {
                S = S.substring(0, i)+
                        S.substring(i+2, S.length());
                flag = true;
                break;
            }
        }
    }
    return S;
}

03 第二种解法

我们还可以借助来实现,借助其后进先出的特性。

思路:遍历S中的字母,如果当前字母和栈顶的字母相等,就将栈顶的字母移除,如果不相等,就压入栈顶,直到遍历完所有字母。最后将栈中的字母拼接到StringBuilder中去,转为字符串时,要反转,因为最先出栈的字母是S中较后面的。

public String removeDuplicates2(String S) {
    Stack<Character> stack = new Stack<Character>();
    int n = S.length();
    for (int i=0; i<n; i++) {
        if (!stack.isEmpty() && S.charAt(i) == stack.peek()) {
            stack.pop();
        } else {
            stack.push(S.charAt(i));
        }
    }
    StringBuilder sb = new StringBuilder();
    while (!stack.isEmpty()) {
        sb.append(stack.pop());
    }
    return sb.reverse().toString();
}

04 第三种解法

同样借助栈的思路,但是我们可以不使用栈,通过StringBuilder来完成类似栈的操作。

思路:创建一个StringBuilder对象,遍历S中的字母,如果当前字母和StringBuilder中的最后一个字母相等,就将StringBuilder中的最后一个字母删除,否则就继续往后拼接。

public String removeDuplicates3(String S) {
    StringBuilder sb = new StringBuilder();
    int n = S.length();
    for (int i=0; i<n; i++) {
        int size = sb.length();
        if (size > 0 && sb.charAt(size-1) == S.charAt(i)) {
            sb.deleteCharAt(size-1);
        } else {
            sb.append(S.charAt(i));
        }
    }
    return sb.toString();
}

05 第四种解法

同样借助栈的思路,我们还可以利用char数组来实现。

思路:创建一个长度和S相等的char数组,遍历S中的字母,如果当前字母和char数组中的最后一个字母相等,就将索引前移,下次再往char数组中添加元素时,就会根据新的索引重新赋值,起到移除重复字母的效果。

public String removeDuplicates4(String S) {
    int n = S.length(), index = 0;
    char[] arr = new char[n];
    for (int i=0; i<n; i++) {
        if (index > 0 && arr[index-1] == S.charAt(i)) {
            index--;
        } else {
            arr[index++] = S.charAt(i);
        }
    }
    return new String(arr, 0, index);
}

06 小结

算法专题目前已连续日更超过七个月,算法题文章257+篇,公众号对话框回复【数据结构与算法】、【算法】、【数据结构】中的任一关键词,获取系列文章合集。

以上就是全部内容,如果大家有什么好的解法思路、建议或者其他问题,可以下方留言交流,点赞、留言、转发就是对我最大的回报和支持!

原文地址:https://www.cnblogs.com/xiaochuan94/p/11223790.html

时间: 2024-11-17 08:19:00

LeetCode.1047-重复删除字符串中的所有相邻重复项的相关文章

[算法]删除字符串中重复的字符

如何删除字符串中重复的字符 问题描述: 删除字符串中重复的字符,例如,"good"去掉重复的字符串后就变成"god". 第一种方法: "蛮力法",最简单的方法就是把这个字符串看作是一个字符数组,对该数组使用双重循环进行遍历,如果发现有重复的字符,就把该字符置为'\0',最后再把这个字符数组中所有的'\0'去掉,此时得到的字符串就是删除重复字符后的目标字符串. 代码如下: package 删除字符串中重复的字符; public class Solu

javascript删除字符串中重复的字符

javascript删除字符串中重复的字符:本章节介绍一下如何删除一个字符串中重复的字符,先不管有没有实际价值,就当做是一种对算法的学习也是挺不错的.代码如下: function dropRepeat(str){ var result=[]; var hash={}; for(var i=0, elem; i<str.length;i++){ elem=str[i]; if(!hash[elem]){ hash[elem]=true; result=result+elem; } } return

C#删除字符串中重复字符的方法

本文实例讲述了C#删除字符串中重复字符的方法.分享给大家供大家参考.具体实现方法如下: 代码如下: #region 删除重复字符 string s = "sdfffffsrlkjesgljfdg03940864e5=_0R9DTGR98YUI\\|||'\\???fdf///"; Response.Write("<br/>String:" + s + "<br/>Result:"); IEnumerable<char

将字符串中连续出现的重复字母进行压缩

原文地址:http://leihuang.net/2014/05/19/List-Interviews/ 单链表的一些常见面试题汇总 单链表反转/逆序 求单链表倒数第N个数 找到单链表的中间结点 如何判断链表是否有环的存在 单链表建环,无环链表变有环 如何知道环的长度? 如何找出环的连接点在哪里? 删除单链表中的重复元素 下面我先简单叙述一下每道题的思路,然后把实现的程序一起贴出来,不会讲得太细,我觉得只要有了思路之后,接下来的难点就是语言上的一些细节问题了,这个不自己去实现,听别人讲是体会不到

字符串-02. 删除字符串中的子串(20)

输入2个字符串S1和S2,要求删除字符串S1中出现的所有子串S2,即结果字符串中不能包含S2. 输入格式: 输入在2行中分别给出不超过80个字符长度的.以回车结束的2个非空字符串,对应S1和S2. 输出格式: 在一行中输出删除字符串S1中出现的所有子串S2后的结果字符串. 输入样例: Tomcat is a male ccatat cat 输出样例: Tom is a male import java.util.Scanner; public class Main { public static

HDU 3518 Boring counting(后缀数组啊 求字符串中不重叠的重复出现至少两次的子串的个数)

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3518 Problem Description 035 now faced a tough problem,his english teacher gives him a string,which consists with n lower case letter,he must figure out how many substrings appear at least twice,moreover

删除字符串中指定位置的字符

/********************************************************************** * 版权所有 (C)2015, Wu Yingqiang. * * 文件名称:DelPosChar.c * 文件标识:无 * 内容摘要:删除字符串中指定位置的字符 * 其它说明:无 * 当前版本: V1.0 * 作 者: Wu Yingqiang * 完成日期: 20150115 * ***********************************

删除字符串中出现次数最少的字符

题目描述 实现删除字符串中出现次数最少的字符,若多个字符出现次数一样,则都删除.输出删除这些单词后的字符串,字符串中其它字符保持原来的顺序. 输入描述: 字符串只包含小写英文字母, 不考虑非法输入,输入的字符串长度小于等于20个字节. 输出描述: 删除字符串中出现次数最少的字符后的字符串. 输入例子: abcdd 输出例子: dd 思路:(我的思路有点绕弯,先记录下来,后面有网友简洁思路),首先把字符串元素存入list中,继续以key-字符,value-出现次数存入map中,然后map按照val

删除字符串中多余的空格

作者 : 卿笃军 今天遇到的一道笔试题,后来百度 了一下,原来是一道新浪的面试题. 题目大意是这样:给你一个任意字符串,要求你删除字符串中多余的空格. 示例:('_'表示空格) 1)"12__abc_98_"     ==> "12_abc_98" 2)"______65_21__54__3_89___"  ==>  "65_21_54_3_89" 思路:设2个下标i,pos一个用于遍历字符串(i),另外一个用于指