oi-wiki
Version:
wiki for OI / ACM-ICPC
38 lines (22 loc) • 1.45 kB
Markdown
## 何为单调栈
顾名思义,单调栈即满足单调性的栈结构。与单调队列相比,其只在一端进行进出。
为了描述方便,以下举例及伪代码以维护一个整数的单调递增栈为例。
## 如何使用单调栈
### 插入
将一个元素插入单调栈时,为了维护栈的单调性,需要在保证将该元素插入到栈顶后整个栈满足单调性的前提下弹出最少的元素。
例如,栈中自顶向下的元素为 ${1,3,5,10,30,50}$,插入元素 $20$ 时为了保证单调性需要依次弹出元素 $1,3,5,10$,操作后栈变为 $20,30,50$。
用伪代码描述如下:
```text
insert x
while !sta.empty() && sta.top()<x
sta.pop()
sta.push(x)
```
### 使用
自然就是从栈顶读出来一个元素,该元素满足单调性的某一端。
例如举例中取出的即栈中的最小值。
## 应用
??? note "[POJ3250 Bad Hair Day](http://poj.org/problem?id=3250)"
有 $N$ 头牛从左到右排成一排,每头牛有一个高度 $h_i$,设左数第 $i$ 头牛与「它右边第一头高度 $≥h_i$」的牛之间有 $c_i$ 头牛,试求 $\sum_{i=1}^{N} c_i$。
(可以左转 [洛谷 P2866](https://www.luogu.org/problemnew/show/P2866))
比较基础的应用有这一题,就是个单调栈的简单应用,记录每头牛被弹出的位置,如果没有被弹出过则为最远端,稍微处理一下即可计算出题目所需结果。