首页 > 其他分享 >单调队列

单调队列

时间:2023-11-29 20:34:23浏览次数:27  
标签:窗口 队列 tt int hh 滑动 单调

一、算法描述

本篇文章讲述的数据结构是单调队列,主要用于解决 滑动窗口 类问题的数据结构,即,在长度为 \(n\) 的序列中,求每个长度为 \(m\) 的区间的区间最值,时间复杂度 \(O(n)\)。

思路如下:

  • 用一个队列 \(q[N]\) 来存储可能是答案的下标。

  • 先判断是否滑出了窗口,如果滑出了则删除队头元素 \(q[hh]\)。

    • \(q[hh]\) 相比于队列中其他元素是最早进来的,所以判断是否在滑动窗口内用 \(q[hh]\) 来判断

      • 如果队列中没有元素,\(i\) 刚好成为 \(q[hh]\)
      • 如果队列中已经存储了元素,\(q[hh]\) 比 \(i\) 早进入队列
      • 所以 \(q[hh]\) 是最早进入队列的
  • 根据单调性,新来的元素如果比之前的元素小,那么只要当前元素存在,则之前的数不可能作为答案,所以可以从队尾出队,直到队尾元素比当前元素还小,或者队列为空

  • 将当前元素加入队列

  • 输出答案

序列中的每个数 \(x\) ,最多只会入队一次、出队一次,所以最多只有 \(2n\) 次操作,最终时间复杂度为 \(O(n)\)。

二、题目描述

给定一个大小为 \(n≤10^6\) 的数组。

有一个大小为 \(k\) 的滑动窗口,它从数组的最左边移动到最右边。

你只能在窗口中看到 \(k\) 个数字。

每次滑动窗口向右移动一个位置。

以下是一个例子:

该数组为 [1 3 -1 -3 5 3 6 7],\(k\) 为 \(3\)。

窗口位置 最小值 最大值
[1 3 -1] -3 5 3 6 7 -1 3
1 [3 -1 -3] 5 3 6 7 -3 3
1 3 [-1 -3 5] 3 6 7 -3 5
1 3 -1 [-3 5 3] 6 7 -3 5
1 3 -1 -3 [5 3 6] 7 3 6
1 3 -1 -3 5 [3 6 7] 3 7

你的任务是确定滑动窗口位于每个位置时,窗口中的最大值和最小值。

输入格式

输入包含两行。

第一行包含两个整数 \(n\) 和 \(k\),分别代表数组长度和滑动窗口的长度。

第二行有 \(n\) 个整数,代表数组的具体数值。

同行数据之间用空格隔开。

输出格式

输出包含两个。

第一行输出,从左至右,每个位置滑动窗口中的最小值。

第二行输出,从左至右,每个位置滑动窗口中的最大值。

输入样例:

8 3
1 3 -1 -3 5 3 6 7 

输出样例:

-1 -3 -3 -3 3 3
3 3 5 5 6 7 

三、题目来源

AcWing算法基础课-154.滑动窗口

四、源代码

#include <iostream>

using namespace std;

const int N = 1000010;

int n, k;
int a[N], q[N];

int main()
{
    cin >> n >> k;
    for (int i = 0; i < n; ++i) cin >> a[i];
    
    int hh = 0, tt = 0;
    for (int i = 0; i < n; ++i)
    {
        if (hh < tt && q[hh] < i - k + 1)   hh ++ ;
        while (hh < tt && a[q[tt - 1]] >= a[i]) tt -- ;
        
        q[tt ++ ] = i;
        
        if (i >= k - 1) cout << a[q[hh]] << ' ';
    }
    puts("");
    
    hh = 0, tt = 0;
    for (int i = 0; i < n; ++i)
    {
        if (hh < tt && q[hh] < i - k + 1)   hh ++ ;
        while (hh < tt && a[q[tt - 1]] <= a[i]) tt -- ;
        
        q[tt ++ ] = i;
        
        if (i >= k - 1) cout << a[q[hh]] << ' ';
    }
    
    return 0;
}

标签:窗口,队列,tt,int,hh,滑动,单调
From: https://www.cnblogs.com/grave-master/p/17865706.html

相关文章

  • 单调栈
    一、算法描述本篇文章讲述的数据结构是单调栈,是一种和单调队列类似的数据结构(下一篇文章会讲到)。单调队列主要用于\(O(n)\)解决滑动窗口问题,单调栈主要用于\(O(n)\)解决NGE问题(NextGreaterElement),也就是对序列中的每个元素,找到上(下)一个比它大(小)的元素,原理相同。以下面......
  • RabbitMQ消息队列
    一.什么是消息队列1.简介在介绍消息队列之前,应该先了解什么是AMQP(AdvancedMessageQueuingProtocol,高级消息队列协议,点击查看)消息(Message)是指在应用间传送的数据,消息可以非常简单,比如只包含文本字符串,也可以更复杂,可能包含嵌入对象;而消息队列(MessageQueue)是一种应用间的......
  • 带有最小间隔时间的队列读取实现 —— 最小等待时间的队列 —— Python编程
     (注:照片源自免费网站,地址:https://www.freepik.com/photos/angry-panda/13)  ==================================================  事情起源是最近在看一个TensorFlow的代码,是TensorFlow实现了一个最小等待时间的队列,解释一下就是一个进程阻塞在一个队列上等待数据的读取,但是......
  • 分段函数的单调性
    分段函数的单调性对高一学生而言是比较难以理解的话题。前言单调性已知分段函数的单调性,求参数的取值范围【高一教学用题】已知\(a\inR\),函数\(f(x)\)满足\(f(x)=\begin{cases}x+3&x\leq1\\x+a&x>1\end{cases}\),函数\(f(x)\)是增函数,求\(a\)解:由题可知,函数的定义域是\(......
  • 支持修改键值的优先队列(以C++,Java为例)
    #include<queue>#include<functional>template<typenameT1,typenameT2>classmutable_priority_queue;template<typenameT1,typenameT2>classmutable_priority_queue{private:std::function<bool(conststd::pair<T1,T......
  • 力扣907. 子数组的最小值之和(单调栈)
    给定一个整数数组 arr,找到 min(b) 的总和,其中 b 的范围为 arr 的每个(连续)子数组。由于答案可能很大,因此 返回答案模 10^9+7 。 示例1:输入:arr=[3,1,2,4]输出:17解释:子数组为[3],[1],[2],[4],[3,1],[1,2],[2,4],[3,1,2],[1,2,4],[3,1,2,4]。最小值为3,1,2,4,1,1,2,1,1,1,和......
  • 单调栈
    一、单调栈简介单调栈(MonotoneStack),拆分一下“单调”,“栈”。也就是说它是在栈的基础之上在多加了一条“单调”的性质。一般来说有单调增加,单调递减两种方式,也就是说从栈顶到栈底,里面的元素是按照一定顺序来排列的。它的时间复杂度为O(n)。二、单调递增栈只有比栈顶元素小的元......
  • 907. 子数组的最小值之和(贡献法,单调栈,前后缀分解)
     题目不难,但是涉及到的知识点很丰富。classSolution:defsumSubarrayMins(self,arr:List[int])->int:MOD=10**9+7n=len(arr)pre=[-1]*nsuf=[n]*nstk=[]foriinrange(n):w......
  • 队列
    一、算法描述本篇文章讲述的数据结构是,队列,数组模拟队列,也不是循环队列。队列的结构,完全就是学校食堂排队打饭的那个队列。一个队头,一个队尾,从队头出,从队尾进,排队打饭也是这样hhh。//用数组模拟的队列定义如下:inthh,tt;intq[N];/* hh表示队头,tt表示队尾(我习惯于表示队尾......
  • 队列(最基本队列,标准队列 2个,双端队列,单调队列)
    2023-11-26最基本队列:一次性使用的classQueue01{//最基本队列,一次性的,数组模拟,先进先出//功能:入队,出队,判满,判空,显示队头,显示队列privateint[]queue;privateintfront=-1;//指向第一个元素前一个位置privateinttail=-1;//指向最后一个元素p......