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

题目

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

在 S 上反复执行重复项删除操作,直到无法继续删除。

在完成所有重复项删除操作后返回最终的字符串。答案保证唯一。

示例 1:

1
2
3
4
输入:"abbaca"
输出:"ca"
解释:
例如,在 "abbaca" 中,我们可以删除 "bb" 由于两字母相邻且相同,这是此时唯一可以执行删除操作的重复项。之后我们得到字符串 "aaca",其中又只有 "aa" 可以执行重复项删除操作,所以最后的字符串为 "ca"。

提示:

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

解法

解法一:

借助栈

JAVA

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
public String removeDuplicates(String S) {
Stack<Character> stack = new Stack<>();
for (char c : S.toCharArray()) {
if (stack.isEmpty()) {
stack.add(c);
continue;
}

if (stack.peek() == c) {
stack.pop();
continue;
}

stack.add(c);
}
StringBuilder sb = new StringBuilder();
while (!stack.isEmpty() ) {
sb.append(stack.pop());
}
return sb.reverse().toString();
}

解法二:

替换函数

  1. aazz 的 26 种重复项放入集合中;
  2. 遍历这 26 种重复项,并用字符串的替换函数把重复项替换成空串。

Java

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public String removeDuplicates(String S) {
// generate 26 possible duplicates
HashSet<String> duplicates = new HashSet();
StringBuilder sb = new StringBuilder();
for (char i = 'a'; i <= 'z'; ++i) {
sb.setLength(0);
sb.append(i); sb.append(i);
duplicates.add(sb.toString());
}

int prevLength = -1;
while (prevLength != S.length()) {
prevLength = S.length();
for (String d : duplicates) S = S.replace(d, "");
}

return S;
}
0%