单调栈的经典例题

简介: 单调栈的经典例题

前言

算法中的单调栈应用十分的广泛;单调栈简单的来说就是栈内元素单调递增或者单调递减的栈,同时单调栈还可以不断的维护一组某种或多种规律的数据,利用这一性质可以解决许多算法问题。本文主要讲解单调栈的算法用法,需对单调栈具有一定的了解。


问题描述

问题一

给你一个字符串 s ,一个整数 k ,一个字母 letter 以及另一个整数 repetition 。返回 s 中长度为 k 字典序最小的子序列,该子序列同时应满足字母 letter 出现至少 repetition 次。生成的测试用例满足 letter s 中出现至少 repetition 次。

子序列是由原字符串删除一些(或不删除)字符且不改变剩余字符顺序得到的剩余字符串。

字符串 a 字典序比字符串 b 小的定义为:在 a b 出现不同字符的第一个位置上,字符串 a 的字符在字母表中的顺序早于字符串 b 的字符。

 

样例:

输入:s = "leetcode", k = 4, letter = "e", repetition= 2

输出:"ecde"

解释:"ecde" 是长度为 4 且满足字母 "e" 出现至少 2 次的字典序最小的子序列。

 

题目来源 力扣:5893. 含特定字母的最小子序列

https://leetcode-cn.com/problems/smallest-k-length-subsequence-with-occurrences-of-a-letter/

题目分析

题意:此题主要描述的是需要维护一个长度为k的字典序最小且至少出现repetition 次letter字母的s的子序列,关键词就是维护一个长度为k的字典序最小的子序列,子序列中包含至少repetition 个letter字母;此题的正好符合单调栈的性质,利用单调栈来维护以上性质!

 

维护三种性质:

性质1:维护字典序单增的字符串。(满足题意中的字典序最小这一性质)

性质2:维护长度为k的字符串。(满足题中长度为k的字符串这一性质)

性质3:维护子序列中包含至少repetition 个letter字母。(同理满足题意性质)

 

const int N = 50010;

class Solution {

public:

    int cnt[N];

    string  smallestSubsequence(string s, int k, char letter, int repetition) {

        int n = s.size();

        for (int i = n-1; i >=  0; i --){

            if (s[i] == letter)  cnt[i] = cnt[i + 1] + 1;

            else cnt[i] = cnt[i+1];  

            //cnt[i]统计在i的右侧还有多少个letter,便于后续维护

        }

        string ans;

        int p = 0; // p用于记录当前维护字符串中的letter数量

 

        // 单调栈模板

        for (int i = 0; i < n; i  ++){

            char c = s[i];

            // 维护字典序单调增的字符串

            while(!ans.empty()  && c < ans.back()){

                // 维护长度为不低于k的字符串

                if (ans.size() - 1  + n - i < k) break;

                if (ans.back() ==  letter){

                    // 维护子序列中包含至少repetition  个letter字母

                    if (p - 1 +  cnt[i] < repetition) break;

                    p --;

                }

                ans.pop_back();

            }

            if (c == letter) p ++,  cnt[i] --;

            ans.push_back(c);

        }

        // 因为最终维护的字符串为长度不低于k的字符串,其长度可能超过k,故此将其长度缩减至k

        while(ans.size() > k) {

            if (ans.back() ==  letter) p --;

            ans.pop_back();

        }

        // 在缩减至k的操作过程中,可能将letter减去,导致字符串中的letter个数少于repetition

        // 所以如果letter个数少于repetition需要将字符串的末尾值换位letter即可满足

        for (int i = ans.size() -  1; i >= 0; i --){

            if (p < repetition  && ans[i] != letter) {

                ans[i] = letter;

                p ++;

            }

        }

        return ans;

    }

};


问题二

返回 s 字典序最小的子序列,该子序列包含 s 的所有不同字符,且只包含一次。

样例

输入:s ="cbacdcbc"

输出:"acdb"

 

题目来源力扣:1081. 不同字符的最小子序列

https://leetcode-cn.com/problems/smallest-subsequence-of-distinct-characters/

题目分析

此题与上一题类似,求一个最小字典序的子序列,且子序列的必须包含s中所有不同的字符只包含一次。

 

维护二种性质:

性质一:维护一个字典序最小的子序列。

性质二:维护包含所有不同字符且只包含一次的子序列。

 

class Solution {

public:

    // cnt记录在s中从i向右中还存在的多少个对应字符

    int cnt[28];

    // snt记录字符在维护的字符串中是否存在

    bool snt[28];

    string  smallestSubsequence(string s) {

        string t;

        int n = s.size(), k = 0;

        for (int i = n-1; i >=  0; i --){

            int x = s[i] - 'a';

            if (cnt[x] == 0) k ++;

            cnt[x] ++;

        }

        for (int i = 0; i < n; i  ++){

            char c = s[i];

            int u = c - 'a';

            // 维护最小字典序的子序列

            while(!t.empty()  && t.back() > c) {

                int x = t.back() -  'a';

                // 维护包含所有不同字符且只包含一次的子序列

                if (cnt[x] <= 0  || snt[u]) break;

                t.pop_back();

                snt[x] = false;

            }

            if (!snt[u])  t.push_back(c);

            snt[u] = true, cnt[u]  --;

        }

        return t;

    }

};


结语

本文主要分享了两道力扣经典的单调栈算法题,其共同的性质都是求最小字典序的子序列,且其余性质均满足单调栈维护;需要对单调栈有了解的朋友进行阅读,了解其单调栈的模板与维护方法。


目录
打赏
0
0
0
0
14
分享
相关文章
|
6天前
|
STL——栈和队列和优先队列
通过以上对栈、队列和优先队列的详细解释和示例,希望能帮助读者更好地理解和应用这些重要的数据结构。
16 4
☀☀☀☀☀☀☀有关栈和队列应用的oj题讲解☼☼☼☼☼☼☼
### 简介 本文介绍了三种数据结构的实现方法:用两个队列实现栈、用两个栈实现队列以及设计循环队列。具体思路如下: 1. **用两个队列实现栈**: - 插入元素时,选择非空队列进行插入。 - 移除栈顶元素时,将非空队列中的元素依次转移到另一个队列,直到只剩下一个元素,然后弹出该元素。 - 判空条件为两个队列均为空。 2. **用两个栈实现队列**: - 插入元素时,选择非空栈进行插入。 - 移除队首元素时,将非空栈中的元素依次转移到另一个栈,再将这些元素重新放回原栈以保持顺序。 - 判空条件为两个栈均为空。
|
2月前
|
【C++数据结构——栈与队列】顺序栈的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现顺序栈的基本运算。开始你的任务吧,祝你成功!​ 相关知识 初始化栈 销毁栈 判断栈是否为空 进栈 出栈 取栈顶元素 1.初始化栈 概念:初始化栈是为栈的使用做准备,包括分配内存空间(如果是动态分配)和设置栈的初始状态。栈有顺序栈和链式栈两种常见形式。对于顺序栈,通常需要定义一个数组来存储栈元素,并设置一个变量来记录栈顶位置;对于链式栈,需要定义节点结构,包含数据域和指针域,同时初始化栈顶指针。 示例(顺序栈): 以下是一个简单的顺序栈初始化示例,假设用C语言实现,栈中存储
158 77
|
2月前
|
C++
【C++数据结构——栈和队列】括号配对(头歌实践教学平台习题)【合集】
【数据结构——栈和队列】括号配对(头歌实践教学平台习题)【合集】(1)遇到左括号:进栈Push()(2)遇到右括号:若栈顶元素为左括号,则出栈Pop();否则返回false。(3)当遍历表达式结束,且栈为空时,则返回true,否则返回false。本关任务:编写一个程序利用栈判断左、右圆括号是否配对。为了完成本关任务,你需要掌握:栈对括号的处理。(1)遇到左括号:进栈Push()开始你的任务吧,祝你成功!测试输入:(()))
43 7
|
2月前
|
【C++数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】
【数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】初始化队列、销毁队列、判断队列是否为空、进队列、出队列等。本关任务:编写一个程序实现环形队列的基本运算。(6)出队列序列:yzopq2*(5)依次进队列元素:opq2*(6)出队列序列:bcdef。(2)依次进队列元素:abc。(5)依次进队列元素:def。(2)依次进队列元素:xyz。开始你的任务吧,祝你成功!(4)出队一个元素a。(4)出队一个元素x。
52 13
【C++数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】
|
2月前
|
【C++数据结构——栈与队列】链栈的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现链栈的基本运算。开始你的任务吧,祝你成功!​ 相关知识 初始化栈 销毁栈 判断栈是否为空 进栈 出栈 取栈顶元素 初始化栈 概念:初始化栈是为栈的使用做准备,包括分配内存空间(如果是动态分配)和设置栈的初始状态。栈有顺序栈和链式栈两种常见形式。对于顺序栈,通常需要定义一个数组来存储栈元素,并设置一个变量来记录栈顶位置;对于链式栈,需要定义节点结构,包含数据域和指针域,同时初始化栈顶指针。 示例(顺序栈): 以下是一个简单的顺序栈初始化示例,假设用C语言实现,栈中存储整数,最大
51 9
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
104 5
|
4月前
|
数据结构之购物车系统(链表和栈)
本文介绍了基于链表和栈的购物车系统的设计与实现。该系统通过命令行界面提供商品管理、购物车查看、结算等功能,支持用户便捷地管理购物清单。核心代码定义了商品、购物车商品节点和购物车的数据结构,并实现了添加、删除商品、查看购物车内容及结算等操作。算法分析显示,系统在处理小规模购物车时表现良好,但在大规模购物车操作下可能存在性能瓶颈。
76 0
|
4月前
|
非递归实现后序遍历时,如何避免栈溢出?
后序遍历的递归实现和非递归实现各有优缺点,在实际应用中需要根据具体的问题需求、二叉树的特点以及性能和空间的限制等因素来选择合适的实现方式。
64 1
数据结构的栈
栈作为一种简单而高效的数据结构,在计算机科学和软件开发中有着广泛的应用。通过合理地使用栈,可以有效地解决许多与数据存储和操作相关的问题。
128 21
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等