洛谷P1440 求m区间内的最小值
题目描述
一个含有n项的数列(n<=2000000),求出每一项前的m个数到它这个区间内的最小值。若前面的数不足m项则从第1个数开始,若前面没有数则输出0。
输入格式:
第一行两个数n,m。
第二行,n个正整数,为所给定的数列。
输出格式:
n行,第i行的一个数ai,为所求序列中第i个数前m个数的最小值。
输入样例:
6 2
7 8 1 4 3 2
输出样例:
0
7
7
1
1
3
数据规模
m ≤ n ≤ 2000000
ai ≤ 3×
1
0
7
10^7
107
题解:
这题第一遍看到就知道是RMQ的题。数据又不会更改,很自然的想到ST表。这样做只能拿80分。还有什么办法———线段树,二分+递归。我要讲一下单调队列的做法。
为了求最小值,我们维护一个单调队列a,如果加进来的数小与队列末的数小,就删去,直到单调队列成升序为止。每次操作也需要删去前m个数之前的数。这样,只要每次的结果,即为单调队列的第一个位置的数值。
代码如下。
/* 2019.2.11
* linwenqi
* luogu1440
* Cpp
*/
#include <iostream>
#include <cstdio>
using namespace std;
int a[2000005][2],head=0,tail=0;
//a[n][0]表示第n个数第几个入队
//a[n][1]表示第n个数的数值 head为头指针 tail为尾指针(最后元素的后一个)
int main(){
int n,m,i,x;
scanf("%d%d",&n,&m);
printf("0\n");//第一个数前没有数
scanf("%d",&a[tail][1]);
tail++;
for(i=1;i<n;i++){
if(i-a[head][0]>m)head++;//单调队列中的元素超过m
printf("%d\n",a[head][1]);//单调队列中最小元素
scanf("%d",&x);
while(a[tail-1][1]>x&&head<tail)
tail--;//单调队列入队操作
a[tail][0]=i;
a[tail][1]=x;
tail++;//注意tail后移
}
return 0;
}