forked from EndlessCheng/codeforces-go
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmonotone_queue.go
279 lines (256 loc) · 8.88 KB
/
monotone_queue.go
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
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
package copypasta
/* 单调队列 Monotone Queue
两张图秒懂单调队列(Python/Java/C++/Go)
https://leetcode.cn/problems/shortest-subarray-with-sum-at-least-k/solution/liang-zhang-tu-miao-dong-dan-diao-dui-li-9fvh/
需要不断维护队列的单调性,时刻保证队列元素从大到小或从小到大
前置知识:双指针
以固定窗口大小的区间最大值为例(此时维护的是一个从大到小的单调队列):
每次向右移动一格左指针,在移动前,如果左指针指向的元素在队首左侧,说明左指针指向的元素小于队首,移动左指针不会改变区间最大值,直接移动左指针即可;
如果左指针指向的就是队首,那么移动左指针会使区间最大值变小(变为单调队列队首之后的那个元素),我们要弹出队首。
这样无论是何种情况,都保证了在移动左指针后,单调队列的队首始终为当前区间的最大值。
https://oi-wiki.org/ds/monotonous-queue/
https://oi-wiki.org/dp/opt/monotonous-queue-stack/
https://cp-algorithms.com/data_structures/stack_queue_modification.html
https://blog.csdn.net/weixin_43914593/article/details/105791217 算法竞赛专题解析(13):DP优化(3)--单调队列优化
todo https://xyzl.blog.luogu.org/DQ-OP-DP
- [面试题 59-II. 队列的最大值](https://leetcode.cn/problems/dui-lie-de-zui-da-zhi-lcof/)(单调队列模板题)
- [239. 滑动窗口最大值](https://leetcode.cn/problems/sliding-window-maximum/)
- [862. 和至少为 K 的最短子数组](https://leetcode.cn/problems/shortest-subarray-with-sum-at-least-k/)
- [1438. 绝对差不超过限制的最长连续子数组](https://leetcode.cn/problems/longest-continuous-subarray-with-absolute-diff-less-than-or-equal-to-limit/)
https://leetcode.cn/tag/monotonic-queue/problemset/
单调队列优化 DP
todo https://www.luogu.com.cn/problem/P2627
http://judge.u-aizu.ac.jp/onlinejudge/description.jsp?id=1070
老鼠进洞 http://codeforces.com/problemset/problem/797/F
LC375 猜数字大小 II https://leetcode-cn.com/problems/guess-number-higher-or-lower-ii/
https://leetcode.cn/problems/guess-number-higher-or-lower-ii/solution/cong-ji-yi-hua-sou-suo-on3-dao-dong-tai-q13g9/
*/
type MqData struct {
Val int
Del int // 懒删除标记
}
type MonotoneQueue struct {
Data []MqData
Size int // 单调队列对应的区间的长度
}
func (mq MonotoneQueue) Less(a, b MqData) bool {
return a.Val >= b.Val // >= 维护区间最大值;<= 维护区间最小值
}
func (mq *MonotoneQueue) Push(v int) {
mq.Size++
d := MqData{v, 1}
for len(mq.Data) > 0 && mq.Less(d, mq.Data[len(mq.Data)-1]) {
d.Del += mq.Data[len(mq.Data)-1].Del
mq.Data = mq.Data[:len(mq.Data)-1]
}
mq.Data = append(mq.Data, d)
}
func (mq *MonotoneQueue) Pop() {
mq.Size--
if mq.Data[0].Del > 1 {
mq.Data[0].Del--
} else {
mq.Data = mq.Data[1:]
}
}
// 返回区间最值
// 调用前需保证 mq.size > 0
func (mq MonotoneQueue) Top() int {
return mq.Data[0].Val
}
// 滑动窗口最值(固定区间大小的区间最值)
// LC239 https://leetcode.cn/problems/sliding-window-maximum/
// https://www.luogu.com.cn/problem/P1886 http://poj.org/problem?id=2823
// https://codeforces.com/problemset/problem/940/E
// https://codeforces.com/problemset/problem/372/C(另一种做法是用堆)
// 贡献+差分数组 https://codeforces.com/problemset/problem/1208/E
func FixedSizeMax(a []int, fixedSize int) []int {
n := len(a)
q := MonotoneQueue{} // 最大/最小由 less 来控制
ans := make([]int, 0, n-fixedSize+1)
for i, v := range a {
q.Push(v)
if q.Size > fixedSize {
q.Pop()
}
// 插入新元素并保证单调队列大小后,获取区间最值
if i+1 >= fixedSize {
ans = append(ans, q.Top())
}
}
return ans
}
// 子区间长度不超过 sizeLimit 的最大子区间和
// 用单调队列维护前缀和的最小值,循环时保证单调队列对应的区间长度不超过 sizeLimit
// https://www.acwing.com/problem/content/137/ https://ac.nowcoder.com/acm/contest/1006/D
func MaxSubSumWithLimitSize(a []int, sizeLimit int) int {
n := len(a)
sum := make([]int, n+1) // int64
for i, v := range a {
sum[i+1] = sum[i] + v
}
ans := int(-1e9) // -1e18
q := MonotoneQueue{} // 维护区间最小值
q.Push(sum[0])
for r := 1; r <= n; r++ {
if q.Size > sizeLimit {
q.Pop()
}
ans = q.max(ans, sum[r]-q.Top())
q.Push(sum[r])
}
return ans
}
// 子数组和至少为 k 的最短非空子数组长度
// 转换成两个前缀和的差至少为 k
// 这题的关键在于,当右端点向右(枚举)时,左端点是绝对不会向左的(因为向左肯定会比当前求出的最短长度要长)
// 想明白这一点就可以愉快地使用单调队列了
// LC862 https://leetcode-cn.com/problems/shortest-subarray-with-sum-at-least-k/
func ShortestSubSumAtLeastK(a []int, k int) int {
n := len(a)
ans := n + 1
sum := make([]int, n+1)
for i, v := range a {
sum[i+1] = sum[i] + v
}
q := MonotoneQueue{} // 维护区间最小值
q.Push(sum[0])
for r := 1; r <= n; r++ {
for q.Size > 0 && sum[r]-q.Top() >= k {
ans = q.min(ans, q.Size)
q.Pop()
}
q.Push(sum[r])
}
if ans > n {
return -1
}
return ans
}
// 对每个右端点,求最远的左端点,满足这一区间内的最大值减最小值不超过 limit
// 求这个的同时,用单调队列维护 DP https://codeforces.com/problemset/problem/487/B
// 完整代码(传入 less 的写法)https://codeforces.com/contest/487/submission/121388184
func LeftPosInDiffLimit(a []int, limit int) []int {
posL := make([]int, len(a))
small := MonotoneQueue{} // 最小值
big := MonotoneQueue{} // 最大值
for i, v := range a {
small.Push(v)
big.Push(v)
for big.Top()-small.Top() > limit {
small.Pop()
big.Pop()
}
posL[i] = i + 1 - small.Size // 通过 size 求出左端点位置
}
return posL
}
// 枚举区间左端点更为方便的情况 · 其一
// 统计区间个数:区间最大值 >= 2*区间最小值
// https://ac.nowcoder.com/acm/contest/6778/C
//
// 思路:转变成求「区间最大值 < 2*区间最小值」的区间个数
// 随着左端点向右,右端点必然不会向左
func CountSubarrayByMinMax(a []int) int {
n := len(a)
ans := n * (n + 1) / 2 // int64
mx := MonotoneQueue{} // 维护区间最大值
mi := MonotoneQueue{} // 维护区间最小值(需要新定义一个有不同 less 的 monotoneQueue)
for i, j := 0, 0; i < n; i++ {
// 确保符合条件再插入
for ; j < n && (mx.Size == 0 || mi.Size == 0 || mx.max(mx.Top(), a[j]) < 2*mi.min(mi.Top(), a[j])); j++ {
mx.Push(a[j])
mi.Push(a[j])
}
sz := j - i
ans -= sz
// 若单调队列指向的区间的左端点为 i,则对应元素在下一次循环时将不再使用。故弹出之
if mx.Size == sz {
mx.Pop()
}
if mi.Size == sz {
mi.Pop()
}
}
return ans
}
// 枚举区间左端点更为方便的情况 · 其二
// https://codeforces.com/problemset/problem/1237/D
// 注意这题和 countSubarrayByMinMax 的不同之处:不满足要求的最小值一定要在最大值的右侧
// 也可以枚举右端点,见 https://www.luogu.com.cn/blog/qianshang/solution-cf1237d
func BalancedPlaylist(a []int, n int) (ans []int) {
a = append(append(a, a...), a...)
q := MonotoneQueue{} // 维护区间最大值
for i, j := 0, 0; i < n; i++ {
// 不断扩大区间右端点 j 直至不满足题目要求
for ; j < 3*n && (q.Size == 0 || q.Top() <= 2*a[j]); j++ {
q.Push(a[j])
}
sz := j - i
if sz > 2*n {
sz = -1
}
ans = append(ans, sz)
if q.Size == sz {
q.Pop()
}
}
return
}
// 二维单调队列
// 输入:一个 n 行 m 列的矩阵 mat
// 输入:高 h 宽 w 的窗口大小
// 返回:一个 n-h+1 行 m-w+1 列的矩阵 areaMax,其中 areaMax[i][j] 表示窗口左上角位于矩阵 (i,j) 时的窗口中元素的最大值
// 例题:HA07 理想的正方形 https://www.luogu.com.cn/problem/P2216
// 解释:https://cdn.acwing.com/media/article/image/2021/06/29/52559_7d7b27ced8-1.png
func FixedSizeAreaMax(mat [][]int, h, w int) [][]int {
n, m := len(mat), len(mat[0])
// 每行求一遍滑窗最值,窗口大小为 w
rowMax := make([][]int, n)
for i := range rowMax {
rowMax[i] = make([]int, m-w+1)
}
for i, row := range mat {
q := MonotoneQueue{} // 维护区间最大值
for j, v := range row {
q.Push(v)
if q.Size > w {
q.Pop()
}
if j+1 >= w {
rowMax[i][j+1-w] = q.Top()
}
}
}
// 对上面的结果再求一遍滑窗最值,窗口大小为 h
areaMax := make([][]int, n-h+1)
for i := range areaMax {
areaMax[i] = make([]int, m-w+1)
}
for j := 0; j+w <= m; j++ {
q := MonotoneQueue{} // 维护区间最大值
for i, mx := range rowMax {
q.Push(mx[j])
if q.Size > h {
q.Pop()
}
if i+1 >= h {
areaMax[i+1-h][j] = q.Top()
}
}
}
return areaMax
}
//
func (MonotoneQueue) min(a, b int) int {
if a > b {
return b
}
return a
}
func (MonotoneQueue) max(a, b int) int {
if b > a {
return b
}
return a
}