【单调栈】LeetCode2030:含特定字母的最小子序列

作者推荐

map|动态规划|单调栈|LeetCode975:奇偶跳

涉及知识点

单调栈

题目

给你一个字符串 s ,一个整数 k ,一个字母 letter 以及另一个整数 repetition 。
返回 s 中长度为 k 且 字典序最小 的子序列,该子序列同时应满足字母 letter 出现 至少 repetition 次。生成的测试用例满足 letter 在 s 中出现 至少 repetition 次。
子序列 是由原字符串删除一些(或不删除)字符且不改变剩余字符顺序得到的剩余字符串。
字符串 a 字典序比字符串 b 小的定义为:在 a 和 b 出现不同字符的第一个位置上,字符串 a 的字符在字母表中的顺序早于字符串 b 的字符。
示例 1:
输入:s = “leet”, k = 3, letter = “e”, repetition = 1
输出:“eet”
解释:存在 4 个长度为 3 ,且满足字母 ‘e’ 出现至少 1 次的子序列:

  • “lee”(“leet”)
  • “let”(“leet”)
  • “let”(“leet”)
  • “eet”(“leet”)
    其中字典序最小的子序列是 “eet” 。
    示例 2:
    输入:s = “leetcode”, k = 4, letter = “e”, repetition = 2
    输出:“ecde”
    解释:“ecde” 是长度为 4 且满足字母 “e” 出现至少 2 次的字典序最小的子序列。
    示例 3:
    输入:s = “bb”, k = 2, letter = “b”, repetition = 2
    输出:“bb”
    解释:“bb” 是唯一一个长度为 2 且满足字母 “b” 出现至少 2 次的子序列。
    参数范围
    1 <= repetition <= k <= s.length <= 5 * 104
    s 由小写英文字母组成
    letter 是一个小写英文字母,在 s 中至少出现 repetition 次

单调栈

最简单的情况:不限制长度和重复字符

从左到右将s[i]放到栈中,放入之前向将栈中大于s[i]的出栈。下面来证明:
假定s的最小子系列的为依次为:{i1,i2,i3…}。
规则一:i1之前没有数小于等于它。否则将此数的放在最前会变得最小。
规则二:i1之后没有数小于它,否则替换i1会变得更小。
规则三:i1和i2之间没有数小于等于i2。
规则四:i2之后没有数小于它,否则替换i2会变得更小。

栈底到栈顶就是最小子系列。
i1在栈底,说明它前面的数都大于它,所以出栈了。
i1没有出栈,说明i1之后没有数比它小,否则i1出栈了。
i2之前,i1之后没有数据,说明两者之间没有数小于等于i2,否则不会出栈。
i2没有出栈说明之后没有数据小于它,否则出栈了。

大数出栈后,变成前面的数小,后面的数大。也就是递增的单调栈。

注意: {1,1,1} 前面增加1,变成{1,1,1,1} 有些规则可能变小,有些规则可能变大。

限制长度k

限制了长度,也就限制了出栈次数。前面的数变小比后面的数变小更小,所以把出栈的机会留给前面。
出栈的时候增加判断:如果出栈后,使得长度一定少于k则不出栈。
入栈的时候增加判断:如果长度已经为k,则不入栈。

必须特定个特定字母

如果出栈会造成特定字母不足则不出栈。
入栈时增加处理:如果长度为k,但当前是特定字母,不入栈则特定字母不足,出栈栈顶所有特定字母,直到非特定字母。出栈它,把出栈的特定字母和当前字母入栈。这个操作在超时的边缘。

代码

核心代码

class Solution {
public:string smallestSubsequence(const string s, const int k, const char letter, const int repetition) {int iCanOutCount = std::count(s.begin(), s.end(), letter) - repetition;//可以不使用letter的数量vector<char> sta;for (int i = 0; i < s.length(); i++){const auto& ch = s[i];auto NeedPop = [&](){if (sta.empty() || (sta.size() + s.length() - i <= k) || (sta.back() <= ch)){return false;}if ((sta.back() == letter) && (iCanOutCount <= 0)){return false;}return true;};while (NeedPop()){iCanOutCount -= (sta.back() == letter);sta.pop_back();}if (sta.size() < k){sta.emplace_back(ch);}else if ((letter == ch)&&( iCanOutCount <= 0 )){int iCnt = 1;while (sta.size()+ iCnt > k ){if (letter == sta.back()){iCnt++;}sta.pop_back();}while (iCnt--){sta.emplace_back(letter);}}else{iCanOutCount -= (letter == ch);}}sta.emplace_back(0);return sta.data();}
};

测试用例

template<class T>
void Assert(const T& t1, const T& t2)
{assert(t1 == t2);
}template<class T>
void Assert(const vector<T>& v1, const vector<T>& v2)
{if (v1.size() != v2.size()){assert(false);return;}for (int i = 0; i < v1.size(); i++){Assert(v1[i], v2[i]);}
}int main()
{string s;int k;char letter;int repetition;{Solution slu;s = "leet", k = 3, letter = 'e', repetition = 1;auto res = slu.smallestSubsequence(s, k, letter, repetition);Assert(std::string("eet"), res);}{Solution slu;s = "leetcode", k = 4, letter = 'e', repetition = 2;auto res = slu.smallestSubsequence(s, k, letter, repetition);Assert(std::string("ecde"), res);}{Solution slu;s = "bb", k = 2, letter ='b', repetition = 2;auto res = slu.smallestSubsequence(s, k, letter, repetition);Assert(std::string("bb"), res);}{Solution slu;s = "aaabbbcccddd", k = 3, letter = 'b', repetition = 2;auto res = slu.smallestSubsequence(s, k, letter, repetition);Assert(std::string("abb"), res);}{Solution slu;s = "hjjhhhmhhwhz", k = 6, letter = 'h', repetition = 5;auto res = slu.smallestSubsequence(s, k, letter, repetition);Assert(std::string("hhhhhh"), res);}//CConsole::Out(res);
}

优化

非特定字母最多入栈:k - repetition。注意 :非特定字母入栈的时候,也要判断栈的长度小于k。

template<class T>
void Assert(const T& t1, const T& t2)
{assert(t1 == t2);
}template<class T>
void Assert(const vector<T>& v1, const vector<T>& v2)
{if (v1.size() != v2.size()){assert(false);return;}for (int i = 0; i < v1.size(); i++){Assert(v1[i], v2[i]);}
}int main()
{string s;int k;char letter;int repetition;{Solution slu;s = "leet", k = 3, letter = 'e', repetition = 1;auto res = slu.smallestSubsequence(s, k, letter, repetition);Assert(std::string("eet"), res);}{Solution slu;s = "leetcode", k = 4, letter = 'e', repetition = 2;auto res = slu.smallestSubsequence(s, k, letter, repetition);Assert(std::string("ecde"), res);}{Solution slu;s = "bb", k = 2, letter ='b', repetition = 2;auto res = slu.smallestSubsequence(s, k, letter, repetition);Assert(std::string("bb"), res);}{Solution slu;s = "aaabbbcccddd", k = 3, letter = 'b', repetition = 2;auto res = slu.smallestSubsequence(s, k, letter, repetition);Assert(std::string("abb"), res);}{Solution slu;s = "hjjhhhmhhwhz", k = 6, letter = 'h', repetition = 5;auto res = slu.smallestSubsequence(s, k, letter, repetition);Assert(std::string("hhhhhh"), res);}//CConsole::Out(res);
}

2023年3月旧版

class Solution {
public:
string smallestSubsequence(string s, int k, const char letter, int repetition) {
m_c = s.length();
int iCanNotUseLetterNum = -repetition;//可以不使用letter多少次
int iCanNotUseChar = m_c - k;
vector vAns;
for (const auto& ch : s)
{
if (ch == letter)
{
iCanNotUseLetterNum++;
}
}
for (int i = 0; i < m_c; i++)
{
const auto& ch = s[i];
while (vAns.size() && (ch < vAns.back()) && iCanNotUseChar && ((letter != vAns.back()) || iCanNotUseLetterNum))
{
if (letter == vAns.back())
{
iCanNotUseLetterNum–;
}
iCanNotUseChar–;
vAns.pop_back();
}
vAns.push_back(ch);
}
string strRet;
for (int iIndex = vAns.size() - 1; iIndex >= 0; iIndex–)
{
const char& ch = vAns[iIndex];
if (0 == iCanNotUseChar)
{
strRet += ch;
continue;
}
if (letter == ch )
{
if (iCanNotUseLetterNum)
{
iCanNotUseLetterNum–;
iCanNotUseChar–;
}
else
{
strRet += ch;
}
}
else
{
iCanNotUseChar–;
}
}
vector tmp(strRet.rbegin(), strRet.rend());
tmp.push_back(0);
return tmp.data();
}
int m_c;
};

扩展阅读

视频课程

有效学习:明确的目标 及时的反馈 拉伸区(难度合适),可以先学简单的课程,请移步CSDN学院,听白银讲师(也就是鄙人)的讲解。
https://edu.csdn.net/course/detail/38771

如何你想快

速形成战斗了,为老板分忧,请学习C#入职培训、C++入职培训等课程
https://edu.csdn.net/lecturer/6176

相关

下载

想高屋建瓴的学习算法,请下载《喜缺全书算法册》doc版
https://download.csdn.net/download/he_zhidan/88348653

我想对大家说的话
闻缺陷则喜是一个美好的愿望,早发现问题,早修改问题,给老板节约钱。
子墨子言之:事无终始,无务多业。也就是我们常说的专业的人做专业的事。
如果程序是一条龙,那算法就是他的是睛

测试环境

操作系统:win7 开发环境: VS2019 C++17
或者 操作系统:win10 开发环境: VS2022 C++17
如无特殊说明,本算法C++ 实现。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/236447.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

SpringMVC01

SpringMVC 1. 学习⽬标2. 什么叫MVC&#xff1f;3. SpringMVC 框架概念与特点4. SpringMVC 请求流程5. Spring MVC 环境搭建6. URL 地址映射配置7. 参数绑定8. JSON 数据开发JSON普通数组步骤1:pom.xml添加依赖步骤2&#xff1a; 修改配置⽂件步骤3. 注解使⽤ 1. 学习⽬标 2. 什…

树莓派,opencv,Picamera2利用舵机云台追踪特定颜色对象(PID控制)

一、需要准备的硬件 Raspiberry 4b两个SG90 180度舵机&#xff08;注意舵机的角度&#xff0c;最好是180度且带限位的&#xff0c;切勿选360度舵机&#xff09;二自由度舵机云台&#xff08;如下图&#xff09;Raspiberry CSI 摄像头 组装后的效果&#xff1a; 二、项目目标…

使用python读取EXCEL放假日历并制作订阅文件

前言 不想升级IOS&#xff0c;苦于找不到新的日历订阅url&#xff0c;小菜鸡百度来百度去发现ics这东西可以自己做一个&#xff0c;惊喜于看到了这篇文章--使用python获取日历信息并制作订阅文件_https: //github.com/lk-itween/calendar-CSDN博客 感谢作者大大。就想自己写一…

服务器数据恢复-昆腾存储StorNext文件系统下raid5数据恢复案例

服务器数据恢复环境&#xff1a; 昆腾某型号存储&#xff0c;StorNext文件存储系统。 共有9个分别配置了24块磁盘的磁盘柜&#xff0c;其中8个磁盘柜存放普通数据&#xff0c;1个磁盘柜存放元数据。 存放元数据的磁盘柜中的24块磁盘组建了8组RAID1阵列和1组4盘RAID10阵列&#…

NCV8460ADR2G在汽车和工业应用中高压侧驱动如何破?

NCV8460ADR2G是一款完全保护的高压侧驱动器&#xff0c;可用于开关各种负载&#xff0c;如灯泡、电磁阀和其他致动器。该器件可以通过有源电流限制和高温关断针对过载情况进行内部保护。 诊断状态输出引脚提供了高温以及开关状态开路负载情况的数字故障指示。 特性&#xff1a;…

22 Vue3中使用v-for遍历对象

概述 使用v-for遍历对象在真实的开发中比较少见&#xff0c;了解即可。 对象我更喜欢统一称之为字典&#xff0c;假如你哪天发现我在某个前端的教程中把对象叫做字典&#xff0c;请你知道这两个是同一个玩意儿。 所谓字典&#xff0c;就是一种key-value类型的结构的统称。 …

队列(C语言版)

一.队列的概念及结构 队列&#xff1a;只允许在一端进行插入数据操作&#xff0c;在另一端进行删除数据操作的特殊线性表&#xff0c;队列具有 先进先出 FIFO(First In First Out) 入队列&#xff1a;进行插入操作的一端称为 队尾 出队列&#xff1a;进行删除操作的一端称为…

网络安全之Linux环境配置及Linux基础知识讲解<三>

目录 一.下载安装Vmware二.下载安装Kali三.Linux目录结构四.Linux文件属性五.文件目录管理六.vim编辑器 一.下载安装Vmware Vmware官网&#xff1a;https://www.vmware.com 二.下载安装Kali Kali包含数百种工具&#xff0c;可用于各种信息安全任务&#xff0c;例如渗透测试、…

vue导出element表格,xlsx和xlsx-style生成xlsx文件并修改样式

1.下载依赖 npm install xlsx --save npm install file-saver --save npm install xlsx-style --save2.先修改xlsx-style的源码&#xff0c;一旦引入xlsx-style则会报错 xlsx-style使用中常见问题及解决办法&#xff1a; xlsx-style使用中常见问题及解决办法-CSDN博客 在\n…

SpringBoot 多环境开发配置文件

在开发过程中&#xff0c;往往开发环境和生产环境需要不同的配置。为了兼容两种运行环境&#xff0c;提高开发效率&#xff0c;可以使用多环境开发配置文件。 配置文件结构大概是这样&#xff1a; application.yml -主启动配置文件&#xff08;用于控制使用哪种环境配…

Java:获取当前线程的线程组

代码示例&#xff1a; package com.thb;public class Demo4 {public static void main(String[] args) {ThreadGroup threadGroup Thread.currentThread().getThreadGroup();System.out.println(threadGroup.getName());} }运行输出&#xff1a;

“2024山西智博会”由中国人工智能学会和省科学技术协会联合主办

近日&#xff0c;山西省政府新闻办近日举行了“山西加快转型发展”系列主题新闻发布会的第六场发布会&#xff0c;同时也是“推动数字经济发展壮大”专场发布会。在发布会上&#xff0c;省委、省政府强调了数字经济的重要性&#xff0c;并将其作为重组要素资源、重塑经济结构、…

【无标题】欢迎使用Markdown编辑器

这里写自定义目录标题 欢迎使用Markdown编辑器新的改变功能快捷键合理的创建标题&#xff0c;有助于目录的生成如何改变文本的样式插入链接与图片如何插入一段漂亮的代码片生成一个适合你的列表创建一个表格设定内容居中、居左、居右SmartyPants 创建一个自定义列表如何创建一个…

第五节TypeScript 运算符

一、描述 运算符用于执行程序代码运算。 二、运算符主要包括&#xff1a; 算术运算符逻辑运算符关系运算符按位运算符赋值运算符三元/条件运算符字符串运算符类型运算符 1、算术运算符 y5&#xff0c;对下面算术运算符进行解释&#xff1a; 运算符 描述 例子 x 运算结果…

ros2 humble安装joint_state_publisher功能包

第一步从GitHub下载此功能包&#xff1a;命令 git clone -b ros2 https://github.com/ros/joint_state_publisher.git 然后执行里面的setup.py文件就完成安装&#xff1b;命令是 sudo python setup.py install 最后检查是不是有安装好&#xff0c;输入命令&#xff1a; ro…

多表插入、删除操作(批量)——后端

多表插入 场景&#xff1a;当添加一个菜品时&#xff0c;还需要记录菜品的口味信息&#xff0c;因此需要对菜品表&#xff08;dish&#xff09;和口味表&#xff08;dish_flavor&#xff09;同时进行插入操作。 两个表的字段&#xff1a; 代码思路&#xff1a;由DishControll…

PSP - 结构生物学中的机器学习 (NIPS MLSB Workshop 2023.12)

欢迎关注我的CSDN&#xff1a;https://spike.blog.csdn.net/ 本文地址&#xff1a;https://spike.blog.csdn.net/article/details/135120094 Machine Learning in Structural Biology (机器学习在结构生物学中) 网址&#xff1a;https://www.mlsb.io/ Workshop at the 37th Co…

重磅发布|2023年度中国可观测性现状调研报告发布

研究目的 随着数智化转型的加速推进&#xff0c;企业的基础架构和应用程序愈发复杂&#xff0c;不确定事件频发&#xff0c;系统变得越来越难以搭建和管理。可观测性作为一种关键技术手段&#xff0c;涵盖了多个技术领域&#xff0c;如日志管理、指标监控、智能告警、根因分析…

usb端口管理软件有哪些(usb端口管理软件)

USB端口管理软件在当今的计算机使用中扮演着越来越重要的角色。随着USB设备的普及&#xff0c;如U盘、移动硬盘、数码相机等&#xff0c;相应的管理软件也变得越来越受欢迎。 1、USB端口管理软件可以帮助用户更好地管理和控制U盘的使用。 在插入U盘时&#xff0c;软件可以自动…

基于CTF探讨Web漏洞的利用与防范

写在前面 Copyright © [2023] [Myon⁶]. All rights reserved. 基于自己之前在CTF中Web方向的学习&#xff0c;总结出与Web相关的漏洞利用方法&#xff0c;主要包括&#xff1a;密码爆破、文件上传、SQL注入、PHP伪协议、反序列化漏洞、命令执行漏洞、文件包含漏洞、Vim…