[算法基础] 栈相关-去除重复字母

题目

给你一个仅包含小写字母的字符串,请你去除字符串中重复的字母,使得每个字母只出现一次。需保证返回结果的字典序最小(要求不能打乱其他字符的相对位置)。

示例 1:

输入: "bcabc"
输出: "abc"

示例 2:

输入: "cbacdcbc"
输出: "acdb"

解题思路

首先我们要明确什么是字典序,字典序就是指从前往后比较两个字符串的方法。

  • 首先比较的是首字符,如果不同,则第一个字符较小的字符串更小
  • 如果第一个字符相同,则依次比较第二个字符,直到能够比较出大小为止

在示例1中,bcabc

  • 字符a在字符串中出现了一次,根据题目的要求,该字符必须保留。
  • bc都出现了两次,很明显,我们选取的为a后面的那两个。

在示例2中cbacdcbc

  • ad 各自出现了一次,必须要选取。
  • cba的前后都出现过,则一定选取a之后的
  • ca后面出现了三次,我们可以列出所有的可能
    • acdb(最小)
    • adcb
    • adbc

我们得出最小字典序的具体过程如下

  • c入栈
  • bc的字典序要小,并且c在后面还会重复出现,则弹出c并且入栈b
  • ab的字典序要小,并且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('')
};

你可能感兴趣的:(算法,字符串,算法,栈,数据结构)