码迷,mamicode.com
首页 > 其他好文 > 详细

滑动窗口

时间:2020-03-15 09:40:25      阅读:43      评论:0      收藏:0      [点我收藏+]

标签:++   clu   const   移动   删掉   出队   name   最大   队列   

# 题意
长度为n的序列,只能看到长度为k的滑动窗口,从数组的最左边移动到最右边,
求出每个滑动窗口的最大值和最小值

# 题解
长度为n的序列,每次从输出k个连续的数中的最大值和最小值
暴力做法,对每一段i,i…i+k-1 逐个进行比较得最大值,时间复杂度为O(n*k),
数据范围大就会超时
优化:
单调队列中存的是下标,用来判断是不是超过区间限定长度
当求窗口中的最小值时,
如果队列中存在两个元素,满足 a[i] >= a[j] 且 i < j,那么无论在什么时候我们都不会取 a[i] 作为最小值了,所以可以直接将 a[i] 删掉;
此时队列中剩下的元素严格单调递增,所以队头就是整个队列中的最小值,可以用 O(1)的时间找到;
为了维护队列的这个性质,我们在往队尾插入元素之前,先将队尾大于等于当前数的元素全部弹出即可;
这样所有数均只进队一次,出队一次,所以时间复杂度是 O(n)的

 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 const int N=1e5+10;
 4 int n,k;
 5 int a[N];
 6 int que[N],tt,hh=1;
 7 int main(){
 8     cin>>n>>k;
 9     for(int i=1;i<=n;i++)
10         cin>>a[i];
11     for(int i=1;i<=n;i++){
12         if(hh<=tt && i-k+1>que[hh]) hh++;
13         while(hh<=tt && a[que[tt]]>=a[i]) tt--;
14         que[++tt]=i;
15         if(i-k+1>0)
16             cout<<a[que[hh]]<< ;
17     }
18     cout<<endl;
19     hh=1,tt=0;
20     for(int i=1;i<=n;i++){
21         if(hh<=tt && i-k+1>que[hh]) hh++;
22         while(hh<=tt && a[que[hh]]<=a[i]) tt--;
23         que[++tt]=i;
24         if(i-k+1>0)
25             cout<<a[que[hh]]<< ;
26     }
27     cout<<endl;
28 }

 

 

滑动窗口

标签:++   clu   const   移动   删掉   出队   name   最大   队列   

原文地址:https://www.cnblogs.com/hhyx/p/12495721.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!