hdu 3530 "Subsequence" (单调队列)
题意:
给出一个序列,求最长的连续子序列,使得 m ≤ Max-Min ≤ k
我的理解:
定义数组 a[] 存储输入的 n 个数;
定义两个双端队列:
deque<int >qMax,qMin;
qMax : 维护前 i 个数的最大值(非递增序列);
qMin : 维护前 i 个数的最小值(非递增序列);
for(int i=;i <= n;++i)
{
///注意此处用了'=',也就是说,队列中的所有数都互异
while(!qMax.empty() && a[qMax.back()] <= a[i])
qMax.pop_back();///保证qMax递减
qMax.push_back(i);
while(!qMin.empty() && a[qMin.back()] >= a[i])
qMin.pop_back();///保证qMin递增
qMin.push_back(i);
}
例如:
1 2 3 4 5 6
val: 9 6 5 1 2 3
i = 6 时的队列情况(此处队列中维护的是值,便于理解):
qMax : {9,6,5,3};
qMin : {1,2,3};
假设来到 i 位置(队列维护下标):
qMax : { φ1,φ2,φ3,...,i };(下标递增,下标对应的值递减)
qMin : { ω1,ω2,ω3,....,i }:(下标递增,下标对应的值递增)

(φ1,φ2,φ3,...ω1,ω2,ω3 互不相同,当然 i 除外)
(qMax.val ≥ qMin.val,下标对应的值)
下面看看队列的操作(pos初始为1):
while(a[qMax.front()]-a[qMin.front()] > k)
{
if(qMax.front() < qMin.front())
{
pos=qMax.front()+;
qMax.pop_front();
}
else
{
pos=qMin.front()+;
qMin.pop_front();
}
}
if(a[qMax.front()]-a[qMin.front()] >= m)
ans=max(ans,i-pos+);
首先判断 qMax.front() 对应的值(假设为 a[φ1]) 与 qMin.front() 对应的值(假设为 a[ω1])做差是否大于 k;
如果大于,那么,如何使差值变小呢?
操作①qMax.pop_front(),(φ1 -> φ2)下一位对应的值更小
操作②qMin.pop_front(),(ω1 -> ω2)下一位对应的值更大
操作①②都可以使他们两者的差值变小,那么,到底该用哪个操作呢?
答案:谁的下标小,弹出谁,并且,所有的出队操作是不可能使队列为空的,因为两个队列中都有a[ i ]这个元素;
为什么要这么做呢?
假设不这么做,那么就是谁的下标大,弹出谁,假设 φ1 > ω1 ;
那么,就需要弹出φ1 ,那么,包含当前最值的区间为[ ω1 ,φ2],但是,φ1 也在其中,最大值就不该是a[φ1 ]而应该是a[φ2];
所以,谁的下标小,弹出谁;
pos作用又是啥呢?
找到包含 a[ i ] 的,并且满足条件的最大的区间,也就是[pos,i]是包含a[i]的最大的区间;
那,为什么pos=front()+1就一定对呢?
假设当前弹出的是φ1 ,也就是说a[φ1]-a[ω1] > t,假设 a[φ2]-a[ω1] ≤ t;
那么,a[pos]-a[ω1] ≤ t;
因为a[pos] ≤ a[φ2],在 φ2 满足条件的情况下,pos一定满足条件,且是满足条件的最大的区间(φ1不满足);
为什么不用判断其是否 ≥ m 呢?
因为操作①②都是使差值减小,只有可能在后面的更新队列的操作中使其差值增大;
AC代码:
#include<iostream>
#include<cstdio>
#include<deque>
using namespace std;
const int maxn=1e5+; int n,m,k;
int a[maxn];
deque<int >qMax,qMin; int Solve()
{
qMax.clear();
qMin.clear(); int ans=;
int pos=;
for(int i=;i <= n;++i)
{
///注意此处用了'=',也就是说,队列中的所有数都互异
///不加'='也行,不影响答案,因为在弹出操作时,qMax,qMin都将等于a[i]的给去了
///那么,a[i]本身是不会更新pos的值
///就算是队列末尾有重复的a[i]也不会更新pos,这就保证了pos的正确性
///[pos,i]是包含a[i]的最大的区间
while(!qMax.empty() && a[qMax.back()] <= a[i])
qMax.pop_back();
qMax.push_back(i);
while(!qMin.empty() && a[qMin.back()] >= a[i])
qMin.pop_back();
qMin.push_back(i); while(a[qMax.front()]-a[qMin.front()] > k)///最坏的情况是最后两个队列只剩下a[i]
{
if(qMax.front() < qMin.front())
{
pos=qMax.front()+;
qMax.pop_front();
}
else
{
pos=qMin.front()+;
qMin.pop_front();
}
}
if(a[qMax.front()]-a[qMin.front()] >= m)
ans=max(ans,i-pos+);///判断[pos,i]区间是否更新ans
}
return ans;
}
int main()
{
while(~scanf("%d%d%d",&n,&m,&k))
{
for(int i=;i <= n;++i)
scanf("%d",a+i);
printf("%d\n",Solve());
}
return ;
}
hdu 3530 "Subsequence" (单调队列)的更多相关文章
- HDU - 3530 Subsequence (单调队列)
Subsequence Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total ...
- hdu 3530 Subsequence 单调队列
题目链接 题目给出n个数, 一个下界m, 一个上界k, 让你求出最长的一段序列, 满足这段序列中的最大的数-最小的数<=k&&>=m, 输出这段长度. 可以维护两个队列, ...
- 【单调队列+尺取】HDU 3530 Subsequence
acm.hdu.edu.cn/showproblem.php?pid=3530 [题意] 给定一个长度为n的序列,问这个序列满足最大值和最小值的差在[m,k]的范围内的最长子区间是多长? [思路] 对 ...
- HDU 3530 Subsequence(单调队列)
传送门 Description There is a sequence of integers. Your task is to find the longest subsequence that s ...
- hdu 3530 Subsequence
题目连接 http://acm.hdu.edu.cn/showproblem.php?pid=3530 Subsequence Description There is a sequence of i ...
- HDU 3401 Trade(单调队列优化)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3401 题意:炒股.第i天买入一股的价钱api,卖出一股的价钱bpi,最多买入asi股,最多卖出bsi股 ...
- Hdu 3410 【单调队列】.cpp
题意: 给出一个数组,问你对于第i个数,从最后一个比它大的数到它之间比它小的数中最大的那个数的下标,以及它右边到第一个比它大的数中比它小的数中最大的那一个数的下标<下标从1开始>. eg: ...
- HDU 5749 Colmerauer 单调队列+暴力贡献
BestCoder Round #84 1003 分析:(先奉上zimpha巨官方题解) 感悟:看到题解单调队列,秒懂如何处理每个点的范围,但是题解的一句算贡献让我纠结半天 已知一个点的up,do ...
- HDU 5289 Assignment(单调队列)
题意:给T足数据,然后每组一个n和k,表示n个数,k表示最大同意的能力差,接下来n个数表示n个人的能力,求能力差在k之内的区间有几个 分析:维护一个区间的最大值和最小值,使得他们的差小于k,于是採用单 ...
随机推荐
- JavaSE从入门到精通
1.JavaSE的安装 windows下安装完成后,配置环境变量如下: JAVA_HOME C:\Program Files (x86)\Java\jdk1.8.0_91 CLASSP ...
- vs code安装
vs code是一款文本编辑器,开源,是前端界的vs,而Dreamweaver适合入门. user版本的一些系统分区文件夹无法创建,可能存在语言显示问题.一般用户建议使用system版. 下载链接:h ...
- Nginx用户权限验证管理
首先需要编译进--with-http_request_model 配置指令:auth_request url | off; #url是指上游服务器地址 context: http/location 备 ...
- Python图形库Turtle
画笔绘制状态函数 函数 描述 pendown() 放下画笔 penup() 提起画笔,与pendown()配合使用 pensize(width) 设置画笔线条的粗细为指定大小 画笔运动函数 函数 描述 ...
- zabbix在ubuntu16.04上的安装
开始安装 zabbix具体安装可以参考官方文档写的很详细,令人高兴的是现在有了中文的版本的翻译,这里简要说下. 上篇文章我写了在ubuntu14.04上安装zabbix,见这里http://www.c ...
- python中的logging模块学习
Python的logging模块 Logging的基本信息: l 默认的情况下python的logging模块打印到控制台,只显示大于等于warning级别的日志 l 日志级别:critical ...
- Sudoku POJ - 3076
Sudoku Time Limit: 10000MS Memory Limit: 65536K Total Submissions: 5769 Accepted: 2684 Descripti ...
- yoj维护
维护 启动容器 docker start yoj 暂停容器 docker stop yoj 重启容器 docker restart yoj 进入容器的终端 docker attach yok 保存容器 ...
- Linux 检查端口gps命令
由于是游戏业务,环境主要是Nginx+Tomcat+Java Program gps脚本环境以及效果图如下: #!/bin/bash function Printf (){ == ];then pri ...
- Educational Codeforces Round 33 (Rated for Div. 2) F. Subtree Minimum Query(主席树合并)
题意 给定一棵 \(n\) 个点的带点权树,以 \(1\) 为根, \(m\) 次询问,每次询问给出两个值 \(p, k\) ,求以下值: \(p\) 的子树中距离 \(p \le k\) 的所有点权 ...