单调栈学习笔记
单调栈是什么
从名字上来看单调栈与单调队列十分的相似,同样的它们的具体内容也十分的相似。单调栈本质上是一个栈,这个栈中的元素是严格有序(即不存在相等元素)的,所以栈顶元素总是比栈里的元素大或小,用公式表达如下:
$$
\forall i \in \left[0, n\right) \begin{cases}
Stack_{i + 1} > Stack_{i}, & NGE \
Stack_{i + 1} < Stack_{i}, & NLE
\end{cases}
$$
其中NGE指的是 Next Greater Element,即下一个比它大的元素,而NLE指的是 Next lower Element,即下一个比它小的元素。
单调栈用来干什么
解决 NGE、NLE 问题。
如何维护单调栈
以 NGE 问题为例。
例题:洛谷 P5788 【模板】单调栈
我们需要获取序列中第$i$个元素之后第一个大于$a_{i}$的元素的下标,联想到单调栈的性质即使用单调栈解决此题。
我们维护一个栈stack
,遍历序列$a$时进行如下操作:
- 检查
stack
是否为空,如果是则将i
入栈并进行下一次判断(continue
)。 - 不断检查
stack
的栈顶元素(下标)在$a$中的取值并且如果$a\left[i\right] > a\left[stack.top\right]$,就将stack.top
的答案设置为i
,因为i
为下一个比它大的元素。
不难发现,经过以上的维护,栈stack
内的元素始终是有序的,所以答案的正确性也就得到了保证。
code:
1 |
|