洛谷 P1440 求m区间内的最小值

洛谷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 
#include 
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;
} 

你可能感兴趣的:(单调队列,单调队列)