
2026-05-25:删除重复字符后的字典序最小字符串。用go语言,给定一个只包含小写字母的字符串 s。你可以重复执行以下操作任意次(也可以不执行):在当前字符串中,挑选一个已经至少出现两次的字母,然后删除它其中一次出现。最后,你需要得到所有可能结果里字典序最小的那个字符串。
字典序比较规则是:从左到右逐位比较字符;一旦某一位不同,就看哪个字符串该位字符在字母表中更靠前,那个字符串就更小;如果前面所有可比较的位都相同,则更短的字符串字典序更小。
1 <= s.length <= 100000。
s 仅包含小写英文字母。
输入: s = "aaccb"。
输出: "aacb"。
解释:
可以形成字符串 "acb"、"aacb"、"accb" 和 "aaccb"。其中 "aacb" 是字典序最小的。
例如,可以选择字母 'c' 并删除它的第一次出现,得到 "aacb"。
题目来自力扣3816。
遍历整个字符串 a a c c b,统计每个小写字母出现的总次数:
a:出现2次c:出现2次b:出现1次left 数组作用:记录当前字符后续还剩多少个未遍历,判断是否可以删除栈顶字符(必须保证删除后,该字符后面还有,才能删)。
栈用于存储最终结果的字符,初始为空。
遍历顺序:a → a → c → c → b,逐个处理:
aa 入栈a 数量减1(left[a] = 1)[a]aa,当前字符也是 a,两者相等a 后续还有剩余(left[a]=1),满足删除条件a,剩余 a 数量再减1(left[a] = 0)a 入栈[a]ca,c > a,无法通过删除栈顶让字典序更小c 入栈c 数量减1(left[c] = 1)[a, c]cc,当前字符也是 c,两者相等c 后续还有剩余(left[c]=1),满足删除条件c,剩余 c 数量再减1(left[c] = 0)c 入栈[a, c]bc,b < c,且栈顶 c 后续已经没有剩余(left[c]=0)c(删除后就没有 c 了,违反每个字符至少保留1次的规则)b 入栈[a, c, b] → 修正:原代码处理后栈为 [a,a,c,b],对应正确结果 aacb遍历完成后,检查栈末尾的字符:
[a,a,c,b] 所有字符都只保留了必要数量,无末尾重复字符可删栈内字符:a, a, c, b
最终字符串:aacb(与题目要求的输出一致)
O(n)2n,耗时 O(n)O(n)n ≤ 100000 大数据量。left 数组:固定大小26,O(1) 常数空间n 个字符,O(n).
package main
import (
"fmt"
)
func lexSmallestAfterDeletion(s string)string {
left := [26]int{}
for _, ch := range s {
left[ch-'a']++
}
st := []rune{}
for _, ch := range s {
// 如果 ch 比栈顶小,移除栈顶,可以让字典序更小
forlen(st) > 0 && ch < st[len(st)-1] && left[st[len(st)-1]-'a'] > 1 {
left[st[len(st)-1]-'a']--
st = st[:len(st)-1]
}
st = append(st, ch)
}
// 最后,移除末尾的重复字母,可以让字典序更小
for left[st[len(st)-1]-'a'] > 1 {
left[st[len(st)-1]-'a']--
st = st[:len(st)-1]
}
returnstring(st)
}
func main() {
s := "aaccb"
result := lexSmallestAfterDeletion(s)
fmt.Println(result)
}

.
# -*-coding:utf-8-*-
def lex_smallest_after_deletion(s: str) -> str:
left = [0] * 26
for ch in s:
left[ord(ch) - ord('a')] += 1
stack = []
for ch in s:
# 如果 ch 比栈顶小,移除栈顶,可以让字典序更小
while stack and ch < stack[-1] and left[ord(stack[-1]) - ord('a')] > 1:
left[ord(stack[-1]) - ord('a')] -= 1
stack.pop()
stack.append(ch)
# 最后,移除末尾的重复字母,可以让字典序更小
while left[ord(stack[-1]) - ord('a')] > 1:
left[ord(stack[-1]) - ord('a')] -= 1
stack.pop()
return''.join(stack)
if __name__ == "__main__":
s = "aaccb"
result = lex_smallest_after_deletion(s)
print(result)
.
#include <iostream>
#include <string>
#include <vector>
#include <stack>
using namespace std;
string lexSmallestAfterDeletion(string s) {
vector<int> left(26, 0);
for (char ch : s) {
left[ch - 'a']++;
}
vector<char> st;
for (char ch : s) {
// 如果 ch 比栈顶小,移除栈顶,可以让字典序更小
while (!st.empty() && ch < st.back() && left[st.back() - 'a'] > 1) {
left[st.back() - 'a']--;
st.pop_back();
}
st.push_back(ch);
}
// 最后,移除末尾的重复字母,可以让字典序更小
while (left[st.back() - 'a'] > 1) {
left[st.back() - 'a']--;
st.pop_back();
}
returnstring(st.begin(), st.end());
}
int main() {
string s = "aaccb";
string result = lexSmallestAfterDeletion(s);
cout << result << endl;
return0;
}
