单调栈详解【C/C++】

简介: 看待一个问题,从不同角度,也许能有不同的收获。

 前言:了解过单调队列后,你会发现单调栈的思想其实挺简单...

当然前提是要了解一下什么是(stack)。

看待一个问题,从不同角度,也许能有不同的收获。

在数学家眼中,单调栈本质上是一个严格或非严格维护的单调递增或单调递减的数学结构。

其核心在于动态的维护动态递增或递减的有序关系。

而对于算法工程师,他们首先关注单调栈的核心优势:O(n)的时间复杂度。在需要遍历序列,并纪录极值的情况下(如接雨水、每日温度),暴力解法通常需要O(n^2),而单调栈,通过每个元素仅入栈与出栈一次,将复杂度,降低至O(n),这对于在大数据的场景下,具有决定性意义。

既然单调栈是一种思想,那必然衍生出了理论(一切操作基于stack)。

单调递增:从栈顶到栈底,保持递增。注意是从栈顶!

================
|6 5 4 3 2 1   <-栈顶
================

image.gif

单调递减:从栈顶到栈底,保持递减。

================
|1 2 3 4 5 6   <-栈顶
================

image.gif

而常见的题型:

  • 下一个更大元素:维护递减栈,当新元素大于栈顶时,栈顶元素的下一个更大元素即为当前元素。
  • 接雨水问题:通过双单调栈分别记录左右边界的最大高度,计算每个位置能接的雨水量。
  • 股票买卖问题:利用单调栈追踪价格趋势,找到最佳买卖点。

接下来,用题目来实操!切记,单调栈内 存入下标更优。

大纲:

1、每日温度--单调栈的简单应用

2、接雨水--用单调栈实现贪心思想 && 能用双指针(对撞指针)替代

题目

1、每日温度

给定一个整数数组 temperatures ,表示每天的温度,返回一个数组 answer ,其中 answer[i] 是指对于第 i 天,下一个更高温度出现在几天后。如果气温在这之后都不会升高,请在该位置用 0 来代替。

示例 1:

输入: temperatures = [73,74,75,71,69,72,76,73]

输出: [1,1,4,2,1,1,0,0]


示例 2:

输入: temperatures = [30,40,50,60]

输出: [1,1,1,0]


示例 3:

输入: temperatures = [30,60,90]

输出: [1,1,0]

提示:

  • 1 <= temperatures.length <= 105
  • 30 <= temperatures[i] <= 100
class Solution {
// 如果暴力,时间复杂度为O(n^2)
// 利用其单调性
public:
    vector<int> dailyTemperatures(vector<int>& temperatures) {
        vector<int> res(temperatures.size());
        stack<int> st;
        for(int i=0; i<temperatures.size(); ++i){
            while(!st.empty()&&temperatures[st.top()]<temperatures[i]){
                res[st.top()]=i-st.top();
                st.pop();
            }
            st.push(i);
        }
        return res;
    }
};

image.gif

2、接雨水

给定 n 个非负整数表示每个宽度为 1 的柱子的高度图,计算按此排列的柱子,下雨之后能接多少雨水。

示例 1:

image.gif 编辑

输入:height = [0,1,0,2,1,0,1,3,2,1,2,1]

输出:6

解释:上面是由数组 [0,1,0,2,1,0,1,3,2,1,2,1] 表示的高度图,在这种情况下,可以接 6 个单位的雨水(蓝色部分表示雨水)。


示例 2:

输入:height = [4,2,0,3,2,5]

输出:9


提示:

  • n == height.length
  • 1 <= n <= 2 * 104
  • 0 <= height[i] <= 105

解题思路:

首先申明一下,本题也能用双指针中的对撞指针解决。

本题巧妙运用单调栈的单调性质 并结合 贪心思想。

设置了一个单调递增栈。

如果有大于栈顶的元素进来了

那代表两个柱子之间可能存在洼地

这时,取栈顶的前两个元素

栈顶top、栈顶的下面的下个元素left

为啥要取top呢,是为了获得水的最低点。

方便计算面积。

class Solution {
public:
    int trap(vector<int>& height) {
        stack<int> st;
        int square = 0;
        for(int i=0; i<height.size(); ++i){
            while(!st.empty() && height[st.top()]<height[i]){
                if(st.size()>=2){
                    int cur = st.top();
                    st.pop();
                    int left = st.top();
                    square += (min(height[left],height[i])-height[cur])*(i-left-1); 
                }else{
                    st.pop();
                }
            }
            st.push(i);
        }
        return square;
    }
};

image.gif


借鉴博客:

1、关于单调栈(Monotone Stack)的详细讲解

2、【算法】一文带你弄懂单调栈!



目录
相关文章
|
1月前
|
文字识别 NoSQL API
Go-Zero微服务实战:高并发场景下的学生认证系统设计与实现
在校园社交等垂直领域应用中,"学生身份认证"是构建信任体系的核心基石。本文将会基于 Go-Zero 微服务框架,详细拆解了一个生产级的学生认证系统实现。涵盖了 OCR 双通道故障转移、WebSocket 实时推送、事件驱动架构 (EDA)、敏感数据加密 以及 有限状态机(FSM) 的设计模式。
196 7
|
1月前
|
存储 人工智能 关系型数据库
OpenClaw怎么可能没痛点?用RDS插件来释放OpenClaw全部潜力
OpenClaw插件是深度介入Agent生命周期的扩展机制,提供24个钩子,支持自动注入知识、持久化记忆等被动式干预。相比Skill/Tool,插件可主动在关键节点(如对话开始/结束)执行逻辑,适用于RAG增强、云化记忆等高级场景。
944 56
OpenClaw怎么可能没痛点?用RDS插件来释放OpenClaw全部潜力
|
1月前
|
消息中间件 NoSQL Redis
高可靠微服务消息设计:Outbox模式、延迟队列与Watermill集成实践
构建高可靠微服务,事件丢失和延迟任务一直是难题?本文带你从实战角度掌握 Outbox模式、延迟队列 及 Watermill+Redis Stream 集成方案,教你用Go打造可靠、可观测、毫秒级响应的事件驱动系统。
185 2
|
1月前
|
缓存 安全 测试技术
GO项目开发规范文档解读
本篇博客的目的,更多是为快速翻阅与回忆使用。
188 1
|
云栖大会 开发者
收到阿里云【乘风者计划】博主证书和奖励
收到阿里云【乘风者计划】博主证书和奖励 2023年2月对我来说是一个很好的开端,因为我在1号就收到了阿里云寄给我的【乘风者计划】博主证书和奖励。好兆头啊! 我收到的是我获得的【技术博主】【星级博主】【专家博主】三个的奖品和证书,一快给我寄过来哒!
3243 2
收到阿里云【乘风者计划】博主证书和奖励
|
1月前
|
安全 索引 Python
5个让你代码更优雅的Python技巧
5个让你代码更优雅的Python技巧
313 133
|
1月前
|
前端开发 JavaScript 应用服务中间件
手把手教你给项目配 HTTPS(Nginx 实战教程,前端 + 后端)
本文章中你既能收获"为什么",也会收获"怎么做"。
368 5
手把手教你给项目配 HTTPS(Nginx 实战教程,前端 + 后端)
|
1月前
|
NoSQL 关系型数据库 MySQL
面向对象的七大设计原则
经艺术设计过的接口,就像蝴蝶一样在指尖翩翩起舞,令人沉醉....
108 0
|
1月前
|
算法
动态规划之完全背包
本文详解完全背包问题:作为动态规划经典题型,区别于01背包(每物限选1次),其特点是每种物品可无限次选取。文章从定义、状态转移方程(dp[i][j] = max(dp[i-1][j], dp[i][j-w]+v))、二维/一维实现到遍历顺序对组合数与排列数的影响,结合零钱兑换II、组合总和IV等5道典型例题深入剖析,助力掌握核心思想与编码技巧。
196 1
|
1月前
|
网络安全 Go Docker
CI/CD全流程
记录 后端go 算法平台 / python 爬虫网关 / 前端vue项目 CI-CD部署流程
280 8