[算法][单调栈] [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

相关文章

C#之List集合去重复对象的实现方法

《C#之List集合去重复对象的实现方法》:本文主要介绍C#之List集合去重复对象的实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C# List集合去重复对象方法1、测试数据2、测试数据3、知识点补充总结C# List集合去重复对象方法1、测试数据

使用C#删除Excel表格中的重复行数据的代码详解

《使用C#删除Excel表格中的重复行数据的代码详解》重复行是指在Excel表格中完全相同的多行数据,删除这些重复行至关重要,因为它们不仅会干扰数据分析,还可能导致错误的决策和结论,所以本文给大家介绍... 目录简介使用工具C# 删除Excel工作表中的重复行语法工作原理实现代码C# 删除指定Excel单元

使用Python和Tkinter实现html标签去除工具

《使用Python和Tkinter实现html标签去除工具》本文介绍用Python和Tkinter开发的HTML标签去除工具,支持去除HTML标签、转义实体并输出纯文本,提供图形界面操作及复制功能,需... 目录html 标签去除工具功能介绍创作过程1. 技术选型2. 核心实现逻辑3. 用户体验增强如何运行

Java如何用乘号来重复字符串的功能

《Java如何用乘号来重复字符串的功能》:本文主要介绍Java使用乘号来重复字符串的功能,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Java乘号来重复字符串的功能1、利用循环2、使用StringBuilder3、采用 Java 11 引入的String.rep

使用雪花算法产生id导致前端精度缺失问题解决方案

《使用雪花算法产生id导致前端精度缺失问题解决方案》雪花算法由Twitter提出,设计目的是生成唯一的、递增的ID,下面:本文主要介绍使用雪花算法产生id导致前端精度缺失问题的解决方案,文中通过代... 目录一、问题根源二、解决方案1. 全局配置Jackson序列化规则2. 实体类必须使用Long封装类3.

SQL常用操作精华之复制表、跨库查询、删除重复数据

《SQL常用操作精华之复制表、跨库查询、删除重复数据》:本文主要介绍SQL常用操作精华之复制表、跨库查询、删除重复数据,这些SQL操作涵盖了数据库开发中最常用的技术点,包括表操作、数据查询、数据管... 目录SQL常用操作精华总结表结构与数据操作高级查询技巧SQL常用操作精华总结表结构与数据操作复制表结

Springboot实现推荐系统的协同过滤算法

《Springboot实现推荐系统的协同过滤算法》协同过滤算法是一种在推荐系统中广泛使用的算法,用于预测用户对物品(如商品、电影、音乐等)的偏好,从而实现个性化推荐,下面给大家介绍Springboot... 目录前言基本原理 算法分类 计算方法应用场景 代码实现 前言协同过滤算法(Collaborativ

MySQL重复数据处理的七种高效方法

《MySQL重复数据处理的七种高效方法》你是不是也曾遇到过这样的烦恼:明明系统测试时一切正常,上线后却频频出现重复数据,大批量导数据时,总有那么几条不听话的记录导致整个事务莫名回滚,今天,我就跟大家分... 目录1. 重复数据插入问题分析1.1 问题本质1.2 常见场景图2. 基础解决方案:使用异常捕获3.

Python实现特殊字符判断并去掉非字母和数字的特殊字符

《Python实现特殊字符判断并去掉非字母和数字的特殊字符》在Python中,可以通过多种方法来判断字符串中是否包含非字母、数字的特殊字符,并将这些特殊字符去掉,本文为大家整理了一些常用的,希望对大家... 目录1. 使用正则表达式判断字符串中是否包含特殊字符去掉字符串中的特殊字符2. 使用 str.isa

openCV中KNN算法的实现

《openCV中KNN算法的实现》KNN算法是一种简单且常用的分类算法,本文主要介绍了openCV中KNN算法的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的... 目录KNN算法流程使用OpenCV实现KNNOpenCV 是一个开源的跨平台计算机视觉库,它提供了各