POJ 2823 Sliding Window(单调队列)

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


你可能感兴趣的:(POJ 2823 Sliding Window(单调队列))