[算法][单调栈] [leetcode]316. 去除重复字母

2024-05-11 22:20

本文主要是介绍[算法][单调栈] [leetcode]316. 去除重复字母,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在这里插入图片描述

  1. 去除重复字母

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

字典序最小:
考虑字符串 a 与 字符串 b,如果字符串 a 在 a 与 b 相异的第一处的字符在字母表上先于对应 b 在此处的字符出现,则称字符串 a 字典序小于 b。
如果 a 或 b 其中较短的字符串为另一个字符串的前半部分,则较短的字符串字典序小于另一个字符串。

示例 1:

输入: s = “bcabc”

输出: “abc”

示例 2:

输入: s = “cbacdcbc”

输出: “acdb”

提示:

1 <= s.length <= 10^4 s由小写英文字母组成

相似题目:该题与1081 https://leetcode-cn.com/problems/smallest-subsequence-of-distinct-characters 相同

思路:

对于s=cbacdcbc,从左到右遍历其中的字母。

1.s[0]=c。由于只遍历了一个字母,目前已知字典序最小的字符串是c。

2.s[1]=b。如果右边没有字母c,那么s0]=c必须保留;实际上右边还有字母c,我们可以
去掉c,改用b当作目前字典序最小的字符串。

3.s[2]=a。同样的,由于右边还有字母b,我们可以去掉b,改用a当作目前字典序最小的字
符串(下面记作am.s)。

4.s[3]=c。由于c比a大,可以接在a后面,目前ans=ac。

5.s[4]=d。由于d比c大,可以接在c后面,目前ans=acd。

6.s[5]=c。由于acd里面已经有c了,直接跳过。目前ans=acd。

7.s[6]=b。我们发现b比d小,能不能像上面s[1]和s2那样,去掉d替换成b呢?这是不
行的,因为后面没有d了,我们只能老老实实地接在d后面,目前ams=acdb。

8.s[7]=c。由于acdb里面已经有c了,直接跳过。

遍历完毕,我们得到了答案ans=acdb。

你可能会问,怎么知首右边是否还有某个字母x?我们可以在遍历s之前,先统计出每个字母的出
现次数,记到一个哈希表或者数组left中。在遍历s时,减少s[i]的出现次数,也就是把left[s[i]]
减一。如果发现left[x]=0就说明右边没有x了。

具体算法如下

1.统计每个字母的出现次数,记到一个哈希表或者数组lfet中。

2.遍历s,先把left[s[i]]减一。

3.如果s[i]在ans中,直接continue。为了快速判断s[i]是否在ams中,可以创建一个哈希
表或者布尔数组inAns。

4.如果s[i]不在ans中,那么判断s[i]是否小于ans的最后一个字母(记作x),如果s[i]<
x且left [x]>0,那么可以把x从ans中去掉,同时标记inAns [x]=false。

5.反复执行第4步,直到ans为空,或者s[i]>x,或者left[x]=0。

6.把s[i]加到ans未尾,同时标记nAns[s[i]]=true。然后继续遍历s的下一个字母。

7.遍历完s后,返回ans。

题解:

    public class Solustion{public String removeDuplicateLetters(String S) {//特殊判断if(null == S){return "";}char [] arr = S.toCharArray();//计算每个字母中出现的次数int [] left = new int[26];for(char c :arr){left[c-'a']++;}StringBuilder ans = new StringBuilder(26);//记录某个字母是否出现过boolean[] inAns = new boolean[26];for(char c : arr){//将次数减少left[c-'a']--;//如果字母已经出现过了则继续if(inAns[c - 'a']){continue;}// 设 x = ans.charAt(ans.length() - 1),// 如果 c < x,且右边还有 x,那么可以把 x 去掉,因为后面可以重新把 x 加到 ans 中while (!ans.isEmpty() && c < ans.charAt(ans.length() - 1) && left[ans.charAt(ans.length() - 1) - 'a'] > 0) {// 标记 x 不在 ans 中inAns[ans.charAt(ans.length() - 1) - 'a'] = false; ans.deleteCharAt(ans.length() - 1);}}return ans.toString();}}

在这里插入图片描述

这篇关于[算法][单调栈] [leetcode]316. 去除重复字母的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/980812

相关文章

SpringBoot实现MD5加盐算法的示例代码

《SpringBoot实现MD5加盐算法的示例代码》加盐算法是一种用于增强密码安全性的技术,本文主要介绍了SpringBoot实现MD5加盐算法的示例代码,文中通过示例代码介绍的非常详细,对大家的学习... 目录一、什么是加盐算法二、如何实现加盐算法2.1 加盐算法代码实现2.2 注册页面中进行密码加盐2.

Java时间轮调度算法的代码实现

《Java时间轮调度算法的代码实现》时间轮是一种高效的定时调度算法,主要用于管理延时任务或周期性任务,它通过一个环形数组(时间轮)和指针来实现,将大量定时任务分摊到固定的时间槽中,极大地降低了时间复杂... 目录1、简述2、时间轮的原理3. 时间轮的实现步骤3.1 定义时间槽3.2 定义时间轮3.3 使用时

C++原地删除有序数组重复项的N种方法

《C++原地删除有序数组重复项的N种方法》给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度,不要使用额外的数组空间,你必须在原地修改输入数组并在使用O(... 目录一、问题二、问题分析三、算法实现四、问题变体:最多保留两次五、分析和代码实现5.1、问题分析5.

如何通过Golang的container/list实现LRU缓存算法

《如何通过Golang的container/list实现LRU缓存算法》文章介绍了Go语言中container/list包实现的双向链表,并探讨了如何使用链表实现LRU缓存,LRU缓存通过维护一个双向... 目录力扣:146. LRU 缓存主要结构 List 和 Element常用方法1. 初始化链表2.

golang字符串匹配算法解读

《golang字符串匹配算法解读》文章介绍了字符串匹配算法的原理,特别是Knuth-Morris-Pratt(KMP)算法,该算法通过构建模式串的前缀表来减少匹配时的不必要的字符比较,从而提高效率,在... 目录简介KMP实现代码总结简介字符串匹配算法主要用于在一个较长的文本串中查找一个较短的字符串(称为

通俗易懂的Java常见限流算法具体实现

《通俗易懂的Java常见限流算法具体实现》:本文主要介绍Java常见限流算法具体实现的相关资料,包括漏桶算法、令牌桶算法、Nginx限流和Redis+Lua限流的实现原理和具体步骤,并比较了它们的... 目录一、漏桶算法1.漏桶算法的思想和原理2.具体实现二、令牌桶算法1.令牌桶算法流程:2.具体实现2.1

Redis 多规则限流和防重复提交方案实现小结

《Redis多规则限流和防重复提交方案实现小结》本文主要介绍了Redis多规则限流和防重复提交方案实现小结,包括使用String结构和Zset结构来记录用户IP的访问次数,具有一定的参考价值,感兴趣... 目录一:使用 String 结构记录固定时间段内某用户 IP 访问某接口的次数二:使用 Zset 进行

Spring Boot 整合 ShedLock 处理定时任务重复执行的问题小结

《SpringBoot整合ShedLock处理定时任务重复执行的问题小结》ShedLock是解决分布式系统中定时任务重复执行问题的Java库,通过在数据库中加锁,确保只有一个节点在指定时间执行... 目录前言什么是 ShedLock?ShedLock 的工作原理:定时任务重复执行China编程的问题使用 Shed

Oracle数据库使用 listagg去重删除重复数据的方法汇总

《Oracle数据库使用listagg去重删除重复数据的方法汇总》文章介绍了在Oracle数据库中使用LISTAGG和XMLAGG函数进行字符串聚合并去重的方法,包括去重聚合、使用XML解析和CLO... 目录案例表第一种:使用wm_concat() + distinct去重聚合第二种:使用listagg,

Python中的随机森林算法与实战

《Python中的随机森林算法与实战》本文详细介绍了随机森林算法,包括其原理、实现步骤、分类和回归案例,并讨论了其优点和缺点,通过面向对象编程实现了一个简单的随机森林模型,并应用于鸢尾花分类和波士顿房... 目录1、随机森林算法概述2、随机森林的原理3、实现步骤4、分类案例:使用随机森林预测鸢尾花品种4.1