天天看点

【洛谷 1886】 滑动窗口 /【模板单调队列】

Time Limit: 12000MS Memory Limit: 65536K

Total Submissions: 89298 Accepted: 24979

Case Time Limit: 5000MS

题目描述

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

数列长度: N < = 1 0 6 , m < = N N<=10^6,m<=N N<=106,m<=N (以最大值为例)

输入格式

输入一共有两行,第一行有两个正整数 n , k n,k n,k。 第二行 n n n 个整数,表示序列 a a a。

输出格式

输出共两行,第一行为每次窗口滑动的最小值

第二行为每次窗口滑动的最大值

输入

8 3

1 3 -1 -3 5 3 6 7

输出

-1 -3 -3 -3 3 3

3 3 5 5 6 7

说明/提示

【数据范围】

对于 50 50 50% 的数据, 1 ≤ n ≤ 1 0 5 ; 1≤n≤10^5 ; 1≤n≤105;

对于 100 100 100% 的数据, 1 ≤ k ≤ n ≤ 1 0 6 , a i ∈ [ − 2 3 1 , 2 3 1 ) 。 1≤k≤n≤10^6,a i ∈[−2^31^ ,2^31^ )。 1≤k≤n≤106,ai∈[−231,231)。

解题思路

【洛谷 1886】 滑动窗口 /【模板单调队列】
【洛谷 1886】 滑动窗口 /【模板单调队列】
【洛谷 1886】 滑动窗口 /【模板单调队列】

分析:我们以求最大值为例

使用单调队列就涉及到去头和删尾:

单调队列具有队列内所有元素不是单调递增就是单调递减的性质,所以每次的最小(最大)值一定会在队首

程序实现过程中先将前 k k k个元素入队,此后每次在队尾加入 a [ k + 1... n ] a[k+1...n] a[k+1...n],在插入元素中同时进行以下操作:

1、将队尾所有大于 a [ i ] a[i] a[i]的值弹出队列

2、插入 a [ i ] a[i] a[i]到队尾

3、判断队首元素位置是否超出 i − k i-k i−k

代码

#include<algorithm>
#include<iostream>
#include<cstdio>
#include<cmath>
using namespace std;
int n,m,a[1000010],h,t;
struct c{
	int x,y;
}st[1000010];
struct cc{
	int x,y;
}st1[1000010];
int main(){
	scanf("%d%d",&n,&m);
	for(int i=1;i<=n;i++)
		scanf("%d",&a[i]);
	h=1,t=1;
	for(int i=1;i<=n;i++)
	{
		while(h<t&&st[t-1].x>=a[i])t--;//当st1[t-1].x比队尾值更优时把队尾值弹出,删尾
		st[t].x=a[i],st[t].y=i,t++;//记录当前点的编号,得到最优解并插入
		if(i>=m)
		{
			while(h<t&&st[h].y<=i-m) h++;//删头
			printf("%d ",st[h].x);	
		}
	}
	printf("\n"); 
	h=1,t=1;
	for(int i=1;i<=n;i++)
	{
		while(h<t&&st1[t-1].x<=a[i])t--;//当st1[t-1].x比队尾值更优时把队尾值弹出,删尾
		st1[t].x=a[i],st1[t].y=i,t++;//记录当前点的编号,得到最优解并插入
		if(i>=m)
		{
			while(h<t&&st1[h].y<=i-m) h++;//删头
			printf("%d ",st1[h].x);	
		}
	}	
	return 0;
} 


           

继续阅读