http://poj.org/problem?id=2823
线段树写9000MS+,实在看不下去了,百度出了单调队列。
这篇介绍不错http://www.cnblogs.com/liukeke/archive/2011/07/31/2122488.html
摘录其维护部分:
如何维护单调队列呢,以单调递增序列为例:
1、如果队列的长度一定,先判断队首元素是否在规定范围内,如果超范围则增长对首。
2、每次加入元素时和队尾比较,如果当前元素小于队尾且队列非空,则减小尾指针,队尾元素依次出队,直到满足队列的调性为止
#include <cstdio>
using namespace std;
const int maxn =1000000+5;
int Max[maxn],Min[maxn],f[maxn],res[maxn],ans[maxn];
int main()
{
int n,m;
while(~scanf("%d%d",&n,&m)){
for(int i=0;i<n;i++) scanf("%d",&f[i]);
int maxfront=0,maxback=0,minfront=0,minback=0;
for(int i=0;i<n;i++)
{
while(f[i]<=f[Min[minback]]&&minback>=minfront) minback--;//维护其单调性
Min[++minback]=i;
while(f[i]>=f[Max[maxback]]&&maxback>=maxfront) maxback--;//同上
Max[++maxback]=i;
while(i-Min[minfront]>=m) minfront++;//维护其长度为小于等于m
while(i-Max[maxfront]>=m) maxfront++;//同上
if(i>=m-1) res[i]=f[Min[minfront]],ans[i]=f[Max[maxfront]];
}
printf("%d",res[m-1]);
for(int i=m;i<n;i++)
printf(" %d",res[i]);
printf("\n");
printf("%d",ans[m-1]);
for(int i=m;i<n;i++)
printf(" %d",ans[i]);
printf("\n");
}
return 0;
}