poj 2823 Sliding Window ( 单调队列 )

11 篇文章 0 订阅
1 篇文章 0 订阅

题意:

给你n个数,然后要你从左到右输出每个区间长度为k的区间上的最小值和最大值。


思路:
这里拿最小值来说,最大值同理。


我们可以这样做
从左往右扫一遍,不断更新最小值,同时还要考虑该最小值是否在当前所考虑的区间里,如果不是的话,就要另找一个合法的最小值
问题是怎么在o(n)或者o(nlogn)的时间内实现。


单调队列刚好可以解决这个问题。
顾名思义,单调队列里的元素都是单调递升(或递减,看需要)的


一开始,初始化队列为空,然后加入一个元素,因为只有一个,所以肯定是递升的
然后,加入第二个,首先判断它与队尾元素的大小关系:
如果大于等于则将其放在队尾,
如果小于则将指向队尾的游标前移,继续判断,直到队尾元素小于该元素,则将其放在队尾,
                                        或者队空,则此时队列中只剩下该元素。
。。。
这样子,队头元素总是保持着最小值。
现在就剩下判断当前最小值是否为合法的了
可以这样做:队列除了存放元素的值之外,还存放它所在的位置。
每次在加入一个元素的时候,同时从队头出发,判断最小值是否合法(当前位置与最小值元素的位置之差小于k),不合法的话就将指向队头的指针后移直到合法为止。
(会不会出现队空呢?是不会的。留给读者思考)


在一开始的时候队列里是空的,所以可以先扫前k-1个元素,让他们根据以上规则进队或出队
然后从第k个元素开始到第n个数,每次把该数加进去,然后去掉不合法的,此时的队头元素就是所求的最小值


总结起来,单调队列无非入队和出队
入队的时候,是从队尾找到第一个小于要入队元素的元素
出队的时候,是从队头开始去除元素,直到队头元素是合法的


下面再以题目中的sample为例,说明单调队列的情况
8 3
1 3 -1 -3 5 3 6 7
首先给每个数都加上它的位置
(1,1)(2,3)(3,-1)(4,-3)(5,5)(6,3)(7,6)(8,7)
一开始的单调队列是空的


先扫前2(即k-1)个元素
队列:(1,1)
3比队尾1大,入队
队列:(1,1) (2,3)


接下来每次加入点都要输出一次
-1比队尾3小,(2,3)出队
-1比队尾1小,(1,1)出队
队列:(3,-1)
输出队头元素


以下不再文字熬述,直接写队列情况(^表示出队)
(3,-1)^
queue:(4,-3)
queue:(4,-3) (5,5)
(5,5)^
queue:(4,-3)(6,3)
queue:(4,-3)(6,3)(7,6)


queue:(4,-3)(6,3)(7,6)(8,7)
(4,-3)^  这里-3出队后才输出队头因为8与4之差已超过了3(即k)


然后,单调队列可以直接用数组模拟,设置两个游标指向队头、队尾就可以了


然后,此题用单调队列写了之后提交,你会发现还是tle。
这不是单调队列的问题,他的复杂度是o(n),因为每次如果去掉的元素的话,虽然会花费比较的时间,但是下一次就不用再比较刚才已经比较过的元素了,所以总的时间复杂度是o(n)
那为什么还会tle呢?

因为数据太强了。。。但还是可以水过去,把编译器从G++改成C++就过了。
这是由于两个编译器对scanf、printf(用cin cout的话更加慢)的编译差别造成的
应该是C++编译器有对C/C++代码进行一些优化的原因

如果非要用G++编译器的话,可以自己写一个putint函数来输出结果,就不会tle了

 

 

#include <cstring>
#include <iostream>
#include <cstdio>

using namespace std;

#define PII pair<int,int>
#define MP(x,y) make_pair(x,y)
#define FI first
#define SE second
const int N=1e6+5;
const int INT_MIN=1<<31;
int a[N],min[N],max[N];
PII que[N];

int front,rear;

inline void putint(int n)
{
    static char buf[20];
    register int pos;
    register int x = n;
    if (x == 0) {
        putchar('0');
        return;
    }
    if (x == INT_MIN) { // x = -x do not work for the minimal value of int, so process it first
        printf("%d", x);
    }
    if (x < 0) {
        putchar('-');
        x = -x;
    }
    pos = 0;
    while (x > 0) {
        buf[pos] = x % 10 + '0';
        x /= 10;
        pos++;
    }
    pos--;
    while (pos >= 0) {
        putchar(buf[pos]);
        pos--;
    }
}

inline void clear()
{
    front=rear=0;
}

inline void dele(int t)
{
   while(front<rear&&que[front].SE<=t)front++;
}

inline void min_insert(PII x)
{
    for(int i=rear-1;i>=front;i--)
    if(que[i].FI<x.FI)
    {
       rear=i+1;
       que[rear++]=x;
       return;
    }
    front=0;
    que[front]=x;
    rear=front+1;
}

inline void max_insert(PII x)
{
    for(int i=rear-1;i>=front;i--)
    if(que[i].FI>x.FI)
    {
        rear=i+1;
        que[rear++]=x;
        return;
    }
    front=0;
    que[front]=x;
    rear=front+1;
}

int main()
{
//    freopen("in","r",stdin);
    int n,k;
    while(scanf("%d%d",&n,&k)>0)
    {
        for(int i=0;i<n;i++)
        scanf("%d",&a[i]);


        clear();
        for(int i=0;i<k-1;i++)
        {
           min_insert(MP(a[i],i));
        }
        for(int i=k-1;i<n-1;i++)
        {
            min_insert(MP(a[i],i));
            dele(i-k);
            //printf("%d ",que[front].FI);
            putint(que[front].FI);
            putchar(' ');
        }
        min_insert(MP(a[n-1],n-1));
        dele(n-1-k);
        //printf("%d\n",que[front].FI);
        putint(que[front].FI);
        putchar('\n');

        clear();
        for(int i=0;i<k-1;i++)
        {
            max_insert(MP(a[i],i));
        }
        for(int i=k-1;i<n-1;i++)
        {
            max_insert(MP(a[i],i));
            dele(i-k);
            //printf("%d ",que[front].FI);
            putint(que[front].FI);
            putchar(' ');
        }
        max_insert(MP(a[n-1],n-1));
        dele(n-1-k);
        //printf("%d\n",que[front].FI);
        putint(que[front].FI);
        putchar('\n');
    }
    return 0;
}

 

 

 

 

 

 

===============

 

以前写的题解、、

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值