单调队列.2

单调队列

什么是单调队列?

单调队列即一个单调递增或递减的队列,例如

1,2,3,4,5

5,4,3,2,1

单调队列有何作用?

单调栈因为是单调的,所以很容易的就能得到他的最大/最小值,一般用来求区间最值。

题目以及代码

给定一个数列,从左至右输出每个长度为m的数列段内的最小数和最大数。 数列长度:N<=106,m<=NN<=106,m<=N 。

简单举例应用
数列为:6 4 10 10 8 6 4 2 12 14
N=10,K=3;
那么我们构造一个长度为3的单调递减队列:
首先,那6和它的位置0放入队列中,我们用(6,0)表示,每一步插入元素时队列中的元素如下
插入6:(6,0);
插入4:(6,0),(4,1);
插入10:(10,2);
插入第二个10,保留后面那个:(10,3);
插入8:(10,3),(8,4);
插入6:(10,3),(8,4),(6,5);
插入4,之前的10已经超出范围所以排掉:(8,4),(6,5),(4,6);
插入2,同理:(6,5),(4,6),(2,7);
插入12:(12,8);
插入14:(14,9);
那么f(i)就是第i步时队列当中的首元素:6,6,10,10,10,10,8,6,12,14
同理,最小值也可以用单调队列来做.

单调队列的时间复杂度是O(N),因为每个数只会进队和出队一次,所以这个算法的效率还是很高的。
注意:建议直接用数组模拟单调队列,因为系统自带容器不方便而且不易调试,同时,每个数只会进去一次,所 以,数组绝对不会爆,空间也是S(N),优于堆或线段树等数据结构。

更重要的:单调是一种思想,当我们解决问题的时候发现有许多冗杂无用的状态时,我们可以采用单调思想,用单 调队列或类似于单调队列的方法去除冗杂状态,保存我们想要的状态.

#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>

/*
1.因为队列是单调的,所以维护的队列中,最后一个数肯定数最大的,跟队列里的数依次比较,直到找到这个数自己的位置。
2.此处数进行替换,因为上一行刚进行了--,往回退一格,现在又++,等于位置没变,新的a[i]找到自己的位置插入。下面for循环的这两句代码也是同理。
*/
using namespace std;
struct node
{
    int x, y;
}v[101]; //x表示值,y表示位置 可以理解为下标

int a[] = { 0,6 ,4, 10, 10 ,8 ,6, 4, 2, 12 ,14 }, n = 10, m = 3, mx[1010000], mn[1010000];//实际从6开始,0只是占位用的。
void getmin()
{
    int i, head = 1, tail = 0;// 默认起始位置为1 因为插入是v[++tail]故初始化为0
    for (i = 1; i < m; i++)//这个循环是先把前m个进入单调队列
    {
        while (head <= tail && v[tail].x >= a[i]) tail--; //1
        v[++tail].x = a[i], v[tail].y = i;//2
        // 根据题目 前m-1个先直接进入队列
    }


    for (; i <= n; i++)//这个循环是先把后n-m个进入单调队列
    {
        while (head <= tail && v[tail].x >= a[i]) tail--;//1 此处如果是有比当前更小的数--
        v[++tail].x = a[i], v[tail].y = i;//2  此处++结合注释1部分,如果是有比当前更小的数,则替换,否走移到下一个位置
        while (v[head].y < i - m + 1) head++;//如果当前头部的下标有更新,则不用移动头部,如果当前头部下标没更新,并且不在范围内,则移动头部++
        mn[i - m + 1] = v[head].x; //把头部的最大数放入结果数组
                                   // 道理同上,当然了 要把已经超出范围的从head开始排出
                                   //  然后每个队首则是目前m个数的最小值
    }
}
int main()
{
    int i;
    getmin();
    for (i = 1; i <= n - m + 1; i++)
    {
        printf(" %d", mn[i]);
    }


    getchar();
    return 0;
}

*

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值