STL-Deque(双端队列)与单调队列的实现
前言:
STl是个好东西,虽然他在不开O2的条件下会跑的很慢,但他着实会让你的代码可读性大大提高,令你的代码看起来既简单又整洁。
双端队列:
顾名思义,双端队列是有两个头的,一个队首指针,一个队尾指针,先进先出或是先进后出都可以实现。
基本操作:
(1) deque<int> dq 定义一个int类型的双端队列dq
(2) deque<int> dq(15) 队列dq具有15个元素单位
(3) deque<int> dq(15,10) 队列dq内15个元素初始值均为10
(4) dq.push_back(x) 将x放入dq的末端
(5) dq.push_front(z) 将x放入dq的前端
(6) dq.size() 返回队列中元素的个数
(7) dq.pop_front() 弹出队列的前端元素
(8) dq.pop_back() 弹出队列的后端元素
(7) dq.front() 返回队列的前端元素
(8) dq.back() 返回队列的后端元素
代码实现:
#include<cstdio>
#include<queue>
using namespace std;
const int maxn=1e5+;
deque<int> dq;
int a[maxn]={,,,,,,,,,,};
int main(){
dq.push_front(a[]),dq.push_front(a[]),dq.push_front(a[]);
dq.push_back(a[]),dq.push_back(a[]),dq.push_back(a[]);
dq.push_front(a[]),dq.push_front(a[]);
dq.push_back(a[]),dq.push_back(a[]);
dq.pop_front(),dq.pop_front();
dq.pop_back(),dq.pop_back();
int n=dq.size();//n=6
for(int i=;i<=n;i++){
printf("%d ",dq.front());
dq.pop_front();
}
return ;
}
deque
小结:
双端队列不仅可以用来优化搜索,更能用来写单调队列这个神奇的xx来优化其他的一些东西(例如DP)
单调队列:
顾名思义,单调队列是单调递增或者是单调递减的一种队列,就像一个递增队列,若将数列1 6 5 3 8放入队列中,则队列的每一步变化为:1、1 6、1 5、1 3、1 3 8 。每当放入的元素使得队列不在单调,则弹出队尾的元素,直到使得队列元素单调。
代码实现:
#include<cstdio>
#include<queue>
using namespace std;
const int maxn=1e5+;
deque<int> q_up;
deque<int> q_dw;
int a[maxn]={,,,,,};
int main(){
for(int i=;i<=;i++){
while(q_up.size()&&q_up.back()>=a[i])
q_up.pop_back();
q_up.push_back(a[i]);
}
int n=q_up.size();
for(int i=;i<=n;i++){
printf("%d ",q_up.front());
q_up.pop_front();
}
return ;
}
q_up
当然,单调队列也可以用来维护滑动窗口的区间最大值和区间最小值。
例题_滑动窗口(洛谷P1886)
题目描述
现在有一堆数字共N个数字(N<=10^6),以及一个大小为k的窗口。现在这个从左边开始向右滑动,每次滑动一个单位,求出每次滑动后窗口中的最大值和最小值。
#include<cstdio>
#include<queue>
using namespace std;
const int maxn=1e6+;
struct cp{
int ord,x;
};
deque<cp> q1,q2;
int n,k;
int a[maxn],s1[maxn],s2[maxn];
inline void q_max(cp e){
while(q1.size()){
cp q=q1.back();
if(q.x>=e.x) q1.pop_back();
else break;
}
while(q1.size()){
cp q=q1.front();
if(q.ord==e.ord-k) q1.pop_front();
else break;
}
q1.push_back(e);
cp q=q1.front();
if(e.ord-k>=) s1[e.ord-k+]=q.x;
}
inline void q_min(cp e){
while(q2.size()){
cp q=q2.back();
if(q.x<=e.x) q2.pop_back();
else break;
}
while(q2.size()){
cp q=q2.front();
if(q.ord==e.ord-k) q2.pop_front();
else break;
}
q2.push_back(e);
cp q=q2.front();
if(e.ord-k>=) s2[e.ord-k+]=q.x;
}
inline int read(){
char ch=getchar();
int x=,f=;
while(ch<''||ch>''){if(ch=='-') f=-;ch=getchar();}
while(ch>=''&&ch<='') x=x*+(ch^),ch=getchar();
return x*f;
}
int main(){
n=read(),k=read();
for(int i=;i<=n;i++) a[i]=read();
for(int i=;i<=n;i++){
cp q;
q.ord=i,q.x=a[i];
q_max(q),q_min(q);
printf("%d ",s1[i])
}
for(int i=;i<=n-k+;i++) ;
putchar('\n');
for(int i=;i<=n-k+;i++) printf("%d ",s2[i]);
return ;
}
Code
结构体类型的队列不知为何不能用 q.front().ord,所以代码显得特别冗长,如果有dalao可以解决这个问题,蒟蒻洗耳恭听!
(以上问题现已解决,附上代码)
#include<cstdio>
#include<queue>
using namespace std;
const int maxn=1e6+;
struct cp{
int ord,x;
};
deque<cp> q1,q2;
int n,k;
int a[maxn],s1[maxn],s2[maxn];
inline void q_max(cp e){
while(q1.size()&&q1.back().x>=e.x) q1.pop_back();
if(q1.size()&&q1.front().ord==e.ord-k) q1.pop_front();
q1.push_back(e);
if(e.ord-k>=) s1[e.ord-k+]=q1.front().x;
}
inline void q_min(cp e){
while(q2.size()&&q2.back().x<=e.x) q2.pop_back();
if(q2.size()&&q2.front().ord==e.ord-k) q2.pop_front();
q2.push_back(e);
if(e.ord-k>=) s2[e.ord-k+]=q2.front().x;
}
inline int read(){
char ch=getchar();
int x=,f=;
while(ch<''||ch>''){if(ch=='-') f=-;ch=getchar();}
while(ch>=''&&ch<='') x=x*+(ch^),ch=getchar();
return x*f;
}
int main(){
n=read(),k=read();
for(int i=;i<=n;i++) a[i]=read();
for(int i=;i<=n;i++){
cp q=(cp){i,a[i]};
q_max(q),q_min(q);
}
for(int i=;i<=n-k+;i++) printf("%d ",s1[i]);
putchar('\n');
for(int i=;i<=n-k+;i++) printf("%d ",s2[i]);
return ;
}
Code(如果您喜欢压行)
如有任何问题,蒟蒻洗耳恭听!
STL-Deque(双端队列)与单调队列的实现的更多相关文章
- [STL] deque 双端队列
- C++STL之双端队列容器
C++STL之双端队列容器 deque双端队列容器与vector很类似,采用线性表顺序存储结构.但与vector区别,deque采用分块的线性存储结构来存储数据,每块的大小一般为512B,将之称为de ...
- deque双端队列容器
//deque双端队列容器 //deque双端队列容器与vector一样,采用线性表顺序存储结构,但与vector不同的是, //deque采用的分块线性存储结构来存储数据,每块的大小一般为512字节 ...
- deque双端队列笔记
clear()clear()clear():清空队列 pushpushpush_back()back()back():从尾部插入一个元素. pushpushpush_front()front()fro ...
- stl之deque双端队列容器
deque与vector很相似,不仅能够在尾部插入和删除元素,还能够在头部插入和删除. 只是当考虑到容器元素的内存分配策略和操作性能时.deque相对vector较为有优势. 头文件 #include ...
- STL容器:deque双端队列学习
所谓deque,是"double-ended queue"的缩写; 它是一种动态数组形式,可以向两端发展,在尾部和头部插入元素非常迅速; 在中间插入元素比较费时,因为需要移动其它元 ...
- 双端队列(单调队列)poj2823 区间最小值(RMQ也可以)
Sliding Window Time Limit: 12000MS Memory Limit: 65536K Total Submissions: 41844 Accepted: 12384 ...
- Java 集合深入理解(10):Deque 双端队列
点击查看 Java 集合框架深入理解 系列, - ( ゜- ゜)つロ 乾杯~ 什么是 Deque Deque 是 Double ended queue (双端队列) 的缩写,读音和 deck 一样,蛋 ...
- c++ deque 双端队列
双端队列: 函数 描述 c.assign(beg,end)c.assign(n,elem) 将[beg; end)区间中的数据赋值给c.将n个elem的拷贝赋值给c. c.at(idx) 传回索引 ...
- 算法-deque双端队列
Python的deque模块,它是collections库的一部分.deque实现了双端队列,意味着你可以从队列的两端加入和删除元素 1.基本介绍 # 实例化一个deque对象d = deque()d ...
随机推荐
- 【Spring注解驱动开发】自定义组件如何注入Spring底层的组件?看了这篇我才真正理解了原理!!
写在前面 最近,很多小伙伴出去面试都被问到了Spring问题,关于Spring,细节点很多,面试官也非常喜欢问一些很细节的技术点.所以,在 Spring 专题中,我们尽量把Spring的每个技术细节说 ...
- JavaScript学习系列博客_28_JavaScript 数组去重
数组去重 var arr = [1,2,3,2,2,1,3,4,2,5]; //去除数组中重复的数字 //获取数组中的每一个元素 for(var i=0 ; i<arr.length ; i++ ...
- 第五篇Scrum冲刺博客--Interesting-Corps
第五篇Scrum冲刺博客 站立式会议 1.会议照片 2.队友完成情况 团队成员 昨日完成 今日计划 鲍鱼铭 音乐详情页面跳转.设计及布局实现设计 搜索页面以及音乐详情页面数据导入及测试 叶学涛 编写分 ...
- 精讲响应式WebClient第5篇-请求超时设置与异常处理
本文是精讲响应式WebClient第5篇,前篇的blog访问地址如下: 精讲响应式webclient第1篇-响应式非阻塞IO与基础用法 精讲响应式WebClient第2篇-GET请求阻塞与非阻塞调用方 ...
- Linux 部署java web 项目,验证码图片不显示文字问题
系统上线后,在获取验证码接口时,获取的验证码图片上没有对应的验证码数字,经过验证后,是由于Linux缺少字体造成的. 正常我们也可以将window的字体直接上传到linux服务器上,window的字体 ...
- 赫然:怎样学习seo优化技术
http://www.wocaoseo.com/thread-79-1-1.html 今天的题目是学习SEO起步阶段每个人都要问的.SEO怎么学?如何进阶SEO技能?都包括哪些知识?笔者也自己总结过一 ...
- 焦大:seo思维光年(中)seo体系化
http://www.wocaoseo.com/thread-56-1-1.html 光年的seo培训一直被业界公认为高端的培训,其主导的seo数据化一直对现在很多人的影响至深,比如我自己.但是也有人 ...
- linux网络配置及虚拟机连接不上网排错思路
第1章 操作系统与虚拟软件的使用 1.1 虚拟软件使用方法 Vmware 1.1.1 开启vmware 注: 同时只能开启一个VMware软件,如果开了两个VMware窗口 提示 ...
- JDBC | 第七章: JDBC数据库连接池使用
概述 数据库连接池是负责分配.管理和释放数据库连接,它允许应用程序重复使用一个现有的数据库连接,而不是再重新建立一个.那么其中的运行机制又是怎样的呢?今天主要介绍一下数据库连接池原理和常用的连接池. ...
- vue前端获取env中的常量
process.env.常量名 如:process.env.MIX_APP_URL