题目
给你一个仅包含小写字母的字符串,请你去除字符串中重复的字母,使得每个字母只出现一次。需保证返回结果的字典序最小(要求不能打乱其他字符的相对位置)。
示例 1:
输入: "bcabc"
输出: "abc"
示例 2:
输入: "cbacdcbc"
输出: "acdb"
解题思路
首先我们要明确什么是字典序,字典序就是指从前往后比较两个字符串的方法。
- 首先比较的是首字符,如果不同,则第一个字符较小的字符串更小
- 如果第一个字符相同,则依次比较第二个字符,直到能够比较出大小为止
在示例1中,bcabc
- 字符
a
在字符串中出现了一次,根据题目的要求,该字符必须保留。 b
和c
都出现了两次,很明显,我们选取的为a
后面的那两个。
在示例2中cbacdcbc
a
和d
各自出现了一次,必须要选取。c
和b
在a
的前后都出现过,则一定选取a
之后的c
在a
后面出现了三次,我们可以列出所有的可能acdb
(最小)adcb
adbc
我们得出最小字典序的具体过程如下
c
入栈b
比c
的字典序要小,并且c
在后面还会重复出现,则弹出c
并且入栈b
a
比b
的字典序要小,并且b
在后面还会重复出现,则弹出b
并且入栈a
c
入栈d
只有一次,直接入栈c
重复出现,不做入栈操作b
入栈c
重复出现,虽然比b
字母序要小,但是题目要求不能改变字符顺序,直接舍弃。
因此,我们在遍历字符串的同时,一定要做两件事
+ 判断字符出现的次数,出现多次的可以依照具体策略舍弃
+ 比较字符大小
该如何舍弃字符呢?我们可以借助栈和计数器,在JavaScript中可以使用indexOf
或者lastIndexOf
。具体策略如下:
+ 遍历字符串里的字符,如果按照正常的ASCII升序,则存到栈中
+ 如果遇到的字符在栈中已经存在,舍弃该字符。
+ 如果读到的字符比栈顶字符小,并且栈顶字符还会出现,则舍弃栈顶字符,存储读取到到的字符。
/**
* @param {string} s
* @return {string}
*/
var removeDuplicateLetters = function(s) {
var stack = []
for(var i = 0;i<s.length;i++){
var char = s[i]
if(stack.indexOf(char) > -1){ continue }
while(stack.length > 0 && stack[stack.length -1 ] > char && s.indexOf(stack[stack.length - 1],i)>i){
stack.pop()
}
stack.push(char)
}
return stack.join('')
};