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; }