单调栈.md 8.4 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
# 如何使用单调栈解题


<p align='center'>
<a href="https://github.com/labuladong/fucking-algorithm" target="view_window"><img alt="GitHub" src="https://img.shields.io/github/stars/labuladong/fucking-algorithm?label=Stars&style=flat-square&logo=GitHub"></a>
<a href="https://www.zhihu.com/people/labuladong"><img src="https://img.shields.io/badge/%E7%9F%A5%E4%B9%8E-@labuladong-000000.svg?style=flat-square&logo=Zhihu"></a>
<a href="https://i.loli.net/2020/10/10/MhRTyUKfXZOlQYN.jpg"><img src="https://img.shields.io/badge/公众号-@labuladong-000000.svg?style=flat-square&logo=WeChat"></a>
<a href="https://space.bilibili.com/14089380"><img src="https://img.shields.io/badge/B站-@labuladong-000000.svg?style=flat-square&logo=Bilibili"></a>
</p>

![](../pictures/souyisou.png)

相关推荐:
  * [回溯算法解题套路框架](https://labuladong.gitbook.io/algo)
  * [动态规划解题套路框架](https://labuladong.gitbook.io/algo)

读完本文,你不仅学会了算法套路,还可以顺便去 LeetCode 上拿下如下题目:

[496.下一个更大元素I](https://leetcode-cn.com/problems/next-greater-element-i)

[503.下一个更大元素II](https://leetcode-cn.com/problems/next-greater-element-ii)

[1118.一月有多少天](https://leetcode-cn.com/problems/number-of-days-in-a-month)

**-----------**
L
labuladong 已提交
26 27 28 29 30 31 32

栈(stack)是很简单的一种数据结构,先进后出的逻辑顺序,符合某些问题的特点,比如说函数调用栈。

单调栈实际上就是栈,只是利用了一些巧妙的逻辑,使得每次新元素入栈后,栈内的元素都保持有序(单调递增或单调递减)。

听起来有点像堆(heap)?不是的,单调栈用途不太广泛,只处理一种典型的问题,叫做 Next Greater Element。本文用讲解单调队列的算法模版解决这类问题,并且探讨处理「循环数组」的策略。

33 34 35 36 37 38 39 40 41 42 43
### 单调栈模板

首先,看一下 Next Greater Number 的原始问题,这是力扣第 496 题「下一个更大元素 I」:

给你一个数组,返回一个等长的数组,对应索引存储着下一个更大元素,如果没有更大的元素,就存 -1。

函数签名如下:

```cpp
vector<int> nextGreaterElement(vector<int>& nums);
```
L
labuladong 已提交
44

45
比如说,输入一个数组 `nums = [2,1,2,4,3]`,你返回数组 `[4,2,4,-1,-1]`
L
labuladong 已提交
46 47 48

解释:第一个 2 后面比 2 大的数是 4; 1 后面比 1 大的数是 2;第二个 2 后面比 2 大的数是 4; 4 后面没有比 4 大的数,填 -1;3 后面没有比 3 大的数,填 -1。

49
这道题的暴力解法很好想到,就是对每个元素后面都进行扫描,找到第一个更大的元素就行了。但是暴力解法的时间复杂度是 `O(n^2)`
L
labuladong 已提交
50 51 52

这个问题可以这样抽象思考:把数组的元素想象成并列站立的人,元素大小想象成人的身高。这些人面对你站成一列,如何求元素「2」的 Next Greater Number 呢?很简单,如果能够看到元素「2」,那么他后面可见的第一个人就是「2」的 Next Greater Number,因为比「2」小的元素身高不够,都被「2」挡住了,第一个露出来的就是答案。

53
![](../pictures/%E5%8D%95%E8%B0%83%E6%A0%88/1.jpeg)
L
labuladong 已提交
54 55 56 57 58

这个情景很好理解吧?带着这个抽象的情景,先来看下代码。

```cpp
vector<int> nextGreaterElement(vector<int>& nums) {
59
    vector<int> res(nums.size()); // 存放答案的数组
L
labuladong 已提交
60
    stack<int> s;
61 62 63 64 65 66
    // 倒着往栈里放
    for (int i = nums.size() - 1; i >= 0; i--) {
        // 判定个子高矮
        while (!s.empty() && s.top() <= nums[i]) {
            // 矮个起开,反正也被挡着了。。。
            s.pop();
L
labuladong 已提交
67
        }
68 69 70 71
        // nums[i] 身后的 next great number
        res[i] = s.empty() ? -1 : s.top();
        // 
        s.push(nums[i]);
L
labuladong 已提交
72
    }
73
    return res;
L
labuladong 已提交
74 75 76
}
```

77 78 79
这就是单调队列解决问题的模板。for 循环要从后往前扫描元素,因为我们借助的是栈的结构,倒着入栈,其实是正着出栈。while 循环是把两个「个子高」元素之间的元素排除,因为他们的存在没有意义,前面挡着个「更高」的元素,所以他们不可能被作为后续进来的元素的 Next Great Number 了。

这个算法的时间复杂度不是那么直观,如果你看到 for 循环嵌套 while 循环,可能认为这个算法的复杂度也是 `O(n^2)`,但是实际上这个算法的复杂度只有 `O(n)`
L
labuladong 已提交
80

81
分析它的时间复杂度,要从整体来看:总共有 `n` 个元素,每个元素都被 `push` 入栈了一次,而最多会被 `pop` 一次,没有任何冗余操作。所以总的计算规模是和元素规模 `n` 成正比的,也就是 `O(n)` 的复杂度。
L
labuladong 已提交
82

83
### 问题变形
L
labuladong 已提交
84

85
单调栈的使用技巧差不多了,来一个简单的变形,力扣第 1118 题「一月有多少天」:
L
labuladong 已提交
86

87 88 89 90 91 92 93
给你一个数组 `T`,这个数组存放的是近几天的天气气温,你返回一个等长的数组,计算:**对于每一天,你还要至少等多少天才能等到一个更暖和的气温;如果等不到那一天,填 0**

函数签名如下:

```cpp
vector<int> dailyTemperatures(vector<int>& T);
```
L
labuladong 已提交
94

95
比如说给你输入 `T = [73,74,75,71,69,76]`,你返回 `[1,1,3,2,1,0]`
L
labuladong 已提交
96

97
解释:第一天 73 华氏度,第二天 74 华氏度,比 73 大,所以对于第一天,只要等一天就能等到一个更暖和的气温,后面的同理。
L
labuladong 已提交
98

99
这个问题本质上也是找 Next Greater Number,只不过现在不是问你 Next Greater Number 是多少,而是问你当前距离 Next Greater Number 的距离而已。
L
labuladong 已提交
100

101
相同的思路,直接调用单调栈的算法模板,稍作改动就可以,直接上代码吧:
L
labuladong 已提交
102 103 104

```cpp
vector<int> dailyTemperatures(vector<int>& T) {
105 106 107 108
    vector<int> res(T.size());
    // 这里放元素索引,而不是元素
    stack<int> s; 
    /* 单调栈模板 */
L
labuladong 已提交
109 110 111 112
    for (int i = T.size() - 1; i >= 0; i--) {
        while (!s.empty() && T[s.top()] <= T[i]) {
            s.pop();
        }
113 114 115 116
        // 得到索引间距
        res[i] = s.empty() ? 0 : (s.top() - i); 
        // 将索引入栈,而不是元素
        s.push(i); 
L
labuladong 已提交
117
    }
118
    return res;
L
labuladong 已提交
119 120 121
}
```

122
单调栈讲解完毕,下面开始另一个重点:如何处理「循环数组」。
L
labuladong 已提交
123

124
### 如何处理环形数组
L
labuladong 已提交
125

126
同样是 Next Greater Number,现在假设给你的数组是个环形的,如何处理?力扣第 503 题「下一个更大元素 II」就是这个问题:
L
labuladong 已提交
127

128
比如输入一个数组 `[2,1,2,4,3]`,你返回数组 `[4,2,4,-1,4]`。拥有了环形属性,**最后一个元素 3 绕了一圈后找到了比自己大的元素 4**
L
labuladong 已提交
129

130
一般是通过 % 运算符求模(余数),来获得环形特效:
L
labuladong 已提交
131 132 133 134 135 136 137 138 139 140

```java
int[] arr = {1,2,3,4,5};
int n = arr.length, index = 0;
while (true) {
    print(arr[index % n]);
    index++;
}
```

141 142 143 144 145
这个问题肯定还是要用单调栈的解题模板,但难点在于,比如输入是 `[2,1,2,4,3]`,对于最后一个元素 3,如何找到元素 4 作为 Next Greater Number。

**对于这种需求,常用套路就是将数组长度翻倍**

![](../pictures/%E5%8D%95%E8%B0%83%E6%A0%88/2.jpeg)
L
labuladong 已提交
146

147
这样,元素 3 就可以找到元素 4 作为 Next Greater Number 了,而且其他的元素都可以被正确地计算。
L
labuladong 已提交
148

149
有了思路,最简单的实现方式当然可以把这个双倍长度的数组构造出来,然后套用算法模板。但是,**我们可以不用构造新数组,而是利用循环数组的技巧来模拟数组长度翻倍的效果**
L
labuladong 已提交
150

151
直接看代码吧:
L
labuladong 已提交
152 153 154 155

```cpp
vector<int> nextGreaterElements(vector<int>& nums) {
    int n = nums.size();
156
    vector<int> res(n);
L
labuladong 已提交
157 158 159
    stack<int> s;
    // 假装这个数组长度翻倍了
    for (int i = 2 * n - 1; i >= 0; i--) {
160
        // 索引要求模,其他的和模板一样
L
labuladong 已提交
161 162 163 164 165 166 167 168 169
        while (!s.empty() && s.top() <= nums[i % n])
            s.pop();
        res[i % n] = s.empty() ? -1 : s.top();
        s.push(nums[i % n]);
    }
    return res;
}
```

170
这样,就可以巧妙解决环形数组的问题,时间复杂度 `O(N)`
L
labuladong 已提交
171

172
如果本文对你有帮助,请三连,这次一定。
L
labuladong 已提交
173

174
**_____________**
L
labuladong 已提交
175

176
**刷算法,学套路,认准 labuladong,公众号和 [在线电子书](https://labuladong.gitbook.io/algo) 持续更新最新文章**
L
labuladong 已提交
177

178
**本小抄即将出版,微信扫码关注公众号,后台回复「小抄」限时免费获取,回复「进群」可进刷题群一起刷题,带你搞定 LeetCode**
L
labuladong 已提交
179

180
<p align='center'>
L
labuladong 已提交
181 182 183 184
<img src="../pictures/qrcode.jpg" width=200 >
</p>

======其他语言代码======