POJ 2823 滑动窗口 单调队列
https://vjudge.net/problem/POJ-2823
中文:https://loj.ac/problem/10175
题目
给一个长度为 $N$ 的数组,一个长为 $K$ 的滑动窗体从最左端移至最右端,你只能看到窗口中的 $K$ 个数,每次窗体向右移动一位,你的任务是找出窗体在各个位置时的最大值和最小值。
题解
滑动窗口模板题……
直接抄紫书上的话……
窗口滑动时,首先删除窗口最左边元素(如果是有用元素),然后把新元素加入单调队列。注意,比新元素大的元素都变得无用了,应当从右往左删除。
举个例子……
窗口初始状态

向右扫描

每次加入新元素到窗口就从右边删掉所有大于这个值的元素

继续……

到这里-1算有用的元素了,会删掉……

$5-2\geqslant k=3$ ,表示这个位置已经不在窗口内了……
为了存位置需要新开一个数组,既然这样干脆窗口里直接存位置就行了,比较的时候通过下标使用就好。
另外在删除的时候可以使用二分确定要删除到的位置,两个循环其实可以合并为一个。
TLE代码(其实用C++交就会AC,用G++才TLE,用快速读写优化就可以AC了)
另外这个代码是主要考虑减少时间使用去了= =显然实际应该考虑三者平衡(时间、空间、编程)……
#include<iostream>
#include<cstdio>
using namespace std;
#define REP(r,x,y) for(register int r=(x); r<(y); r++)
#define REPE(r,x,y) for(register int r=(x); r<=(y); r++)
#ifdef sahdsg
#define DBG(...) printf(__VA_ARGS__)
#else
#define DBG(...)
#endif #define MAXN 1000007 int arr[MAXN], que[MAXN], l, r;
int n,k; int main() {
#ifdef sahdsg
freopen("in.txt", "r", stdin);
#endif
scanf("%d%d", &n, &k);
REP(i,0,n) {
scanf("%d", &arr[i]);
}
l=0, r=0;
REP(i,0,k-1) {
while(l<r && arr[que[r-1]]>arr[i]) r--;
que[r++] = i;
}
bool fi = false;
REP(i,k-1,n) {
if(que[l]<=i-k) l++;
while(l<r && arr[que[r-1]]>arr[i]) r--;
que[r] = i;
if(fi) putchar(' '); else fi = true;
printf("%d", arr[que[l]]);
r++;
}
putchar('\n'); fi = false;
l=0, r=0;
REP(i,0,k-1) {
while(l<r && arr[que[r-1]]<arr[i]) r--;
que[r++] = i;
}
REP(i,k-1,n) {
if(que[l]<=i-k) l++;
while(l<r && arr[que[r-1]]<arr[i]) r--;
que[r] = i;
if(fi) putchar(' '); else fi = true;
printf("%d", arr[que[l]]);
r++;
}
return 0;
}
AC代码:
#include<iostream>
#include<cstdio>
using namespace std;
#define REP(r,x,y) for(register int r=(x); r<(y); r++)
#define REPE(r,x,y) for(register int r=(x); r<=(y); r++)
#ifdef sahdsg
#define DBG(...) printf(__VA_ARGS__)
#else
#define DBG(...)
#endif #define MAXN 1000007 int arr[MAXN], que[MAXN], l, r;
int n,k; template <class T>
inline void read(T& x) {
char c=getchar();int f=1;x=0;
while(!isdigit(c)&&c!='-')c=getchar();if(c=='-')f=-1,c=getchar();
while(isdigit(c)){x=x*10+c-'0';c=getchar();}x*=f;
}
void write(int x) {
if(x<0) {putchar('-');write(-x);return;}
if(x>9) write(x/10); putchar(x%10+'0');
} int main() {
#ifdef sahdsg
freopen("in.txt", "r", stdin);
#endif
read(n); read(k);
REP(i,0,n) {
read(arr[i]);
}
l=0, r=0;
REP(i,0,k-1) {
while(l<r && arr[que[r-1]]>arr[i]) r--;
que[r++] = i;
}
bool fi = false;
REP(i,k-1,n) {
if(que[l]<=i-k) l++;
if(l<r && arr[que[r-1]]>arr[i]) {
int fl=l, fr=r;
while(fr>fl) {
int mid=fl+((fr-fl)>>1); //[l, m) [m,r)
if(arr[que[mid]]>arr[i]) {
fr=mid;
} else {
fl=mid+1;
}
}
r=fr;
}
que[r++] = i;
if(fi) putchar(' '); else fi = true;
write(arr[que[l]]);
}
putchar('\n'); fi = false;
l=0, r=0;
REP(i,0,k-1) {
while(l<r && arr[que[r-1]]<arr[i]) r--;
que[r++] = i;
}
REP(i,k-1,n) {
if(que[l]<=i-k) l++;
if(l<r && arr[que[r-1]]<arr[i]) {
int fl=l, fr=r;
while(fr>fl) {
int mid=fl+((fr-fl)>>1); //[l, m) [m,r)
if(arr[que[mid]]<arr[i]) {
fr=mid;
} else {
fl=mid+1;
}
}
r=fr;
}
que[r++] = i;
if(fi) putchar(' '); else fi = true;
write(arr[que[l]]);
}
return 0;
}
POJ 2823 滑动窗口 单调队列的更多相关文章
- POJ 2823 滑动窗口 单调队列模板
我们从最简单的问题开始: 给定一个长度为N的整数数列a(i),i=0,1,...,N-1和窗长度k. 要求: f(i) = max{a(i-k+1),a(i-k+2),..., a(i)},i = 0 ...
- POJ 2823 Sliding Window + 单调队列
一.概念介绍 1. 双端队列 双端队列是一种线性表,是一种特殊的队列,遵守先进先出的原则.双端队列支持以下4种操作: (1) 从队首删除 (2) 从队尾删除 (3) 从队尾插入 (4) ...
- poj 2823 Sliding Window (单调队列入门)
/***************************************************************** 题目: Sliding Window(poj 2823) 链接: ...
- luoguP1886 滑动窗口 [单调队列]
题目描述 现在有一堆数字共N个数字(N<=10^6),以及一个大小为k的窗口.现在这个从左边开始向右滑动,每次滑动一个单位,求出每次滑动后窗口中的最大值和最小值. 例如: The array i ...
- [POJ2823]Sliding Window 滑动窗口(单调队列)
题意 刚学单调队列的时候做过 现在重新做一次 一个很经典的题目 现在有一堆数字共N个数字(N<=10^6),以及一个大小为k的窗口.现在这个从左边开始向右滑动,每次滑动一个单位,求出每次滑动后窗 ...
- [洛谷P1886]滑动窗口 (单调队列)(线段树)
---恢复内容开始--- 这是很好的一道题 题目描述: 现在有一堆数字共N个数字(N<=10^6),以及一个大小为k的窗口. 现在这个从左边开始向右滑动,每次滑动一个单位,求出每次滑动后窗口中的 ...
- [Luogu P1886]滑动窗口--单调队列入门
题目描述 现在有一堆数字共N个数字(N<=10^6),以及一个大小为k的窗口.现在这个从左边开始向右滑动,每次滑动一个单位,求出每次滑动后窗口中的最大值和最小值. 例如: The array i ...
- POJ 2823 (滑动窗口)
这道题最容易想到的是用朴素的做法,即 每滑动一次,就遍历一次窗口找出最大最小值,这样时间复杂度为O(n*k),由于题目数据比较大,这种做法肯定是超时的. 另外,根据书上的讲解,还可以采用优先队列来求解 ...
- AcWing 154. 滑动窗口 单调队列
地址 https://www.acwing.com/problem/content/description/156/ 输入格式 输入包含两行. 第一行包含两个整数n和k,分别代表数组长度和滑动窗口的长 ...
随机推荐
- mac 下利用AndroidStudio APK获取签名信息
注:网上没找到特别好的.不是xxx.keystore 就是debug.keystore 而真正去找这些目录的时候系统就会提示没有这个秘钥库文件.所以就悲剧了 下面附上快速查看APK签名信息的方法(SH ...
- [Luogu4916]魔力环[Burnside引理、组合计数、容斥]
题意 题目链接 分析 sπo yyb 代码 #include<bits/stdc++.h> using namespace std; typedef long long LL; #defi ...
- centos6.5 squid安装
squid作用 1正向代理 标准的代理缓冲服务器,须在每一个内部主机的浏览器上明确指明代理服务器的IP地址和端口号. 透明代理缓冲服务器,代理操作对客户端的浏览器是透明的(即不需指明代理服务器的IP和 ...
- vue项目打包问题
使用vue-cli脚手架构建vue项目 vue init webpack project npm run build 打包时出现 Tip: built files are meant to be se ...
- koa2实现session的两种方式(基于Redis 和MySQL)
一.基于MySQL的实现方式 这种方式需要安装koa-session-minimal和koa-mysql-session两个依赖. 执行 npm install koa-session-minimal ...
- H5 25-CSS三大特性之层叠性
25-CSS三大特性之层叠性 我是段落 <!DOCTYPE html> <html lang="en"> <head> <meta cha ...
- FPGA中边沿触发和电平触发
边沿触发和电平触发基本就是触发器和锁存器的区别: 触发器是边沿触发,只有当时钟上升(或下降)的一瞬间,触发器会读取并锁存输入信号.输出信号仅在时钟信号上升(或下降)的一瞬间会发生变化. 锁存器是电 ...
- 使用junit测试
package creeper; import java.util.Scanner; public class size { private static int intercePosition = ...
- Linux 安装软件之后设置PATH环境变量
每一个软件都有安装路径这一项,指定安装路径的目的,一方面是便于文件搜索与查找,另一方面更方便的使用软件. 比如,几乎大多数自己安装的软件,都会选择安装在/usr/local目录下,比如apache.m ...
- centos6 yum 安装memcached
centos6 yum 安装memcached - 像块石头 - 博客园http://www.cnblogs.com/rockee/archive/2012/08/01/2619160.html yu ...