zstu 4237 马里奥的求救——(单调队列DP)
题目链接:http://oj.acm.zstu.edu.cn/JudgeOnline/problem.php?id=4237
这题可以转化为每次可以走g~d+x步,求最大分数,且最大分数的步数最少。
这题的数据范围比较小,可以用奇怪的姿势过。
首先,lyf队长给的方法是n^3的dp过;用我自己的方法是搜索也可以过,因为数据小。
但是,如果数据范围很大,就得用复杂度是O(n)的单调队列dp来做。
上次做过一道单调队列的dp问题,当时比较懵懂,现在,对这个方法有了更深的理解。而且,只要把id丢进单调队列里即可(因为可以通过id查找dp的值)。具体过程还是看代码仔细理解吧:
#include <stdio.h>
#include <algorithm>
#include <queue>
#include <set>
using namespace std;
typedef pair<int,int> pii; const int inf =0x3f3f3f3f;
int n,x,g,d;
int a[+];
pii dp[+]; pii solve()
{
deque<int> Q;
dp[]=pii(,);
for(int i=;i<=n;i++)
{
while(!Q.empty() && Q.front()<i-d) Q.pop_front();
if(i-g>= && dp[i-g].first != -inf)
{
while(!Q.empty() && dp[Q.back()] <= dp[i-g]) Q.pop_back();
Q.push_back(i-g);
}
if(!Q.empty()) dp[i] = pii(dp[Q.front()].first + a[i],dp[Q.front()].second - );
//因为题目要求最大分数的最小步数,而对pair的max是默认取大的值,
//所以取个反就是最大化(-步数)
else dp[i] = pii(-inf,);
}
pii ret = pii(-inf,);
for(int i=;i<=n;i++)
{
if(i>=n-d+) ret=max(ret,dp[i]);
//如果处于再跳一步就可以出去的状态的话
}
ret.second = -ret.second + ;
//加1是细节,因为还需要一步跳出
return ret;
} int main()
{
int T;
scanf("%d",&T);
while(T--)
{
scanf("%d%d%d%d",&n,&x,&g,&d);
d += x;
for(int i=;i<=n;i++) scanf("%d",a+i);
pii ans = solve();
printf("%d %d\n",ans.first,ans.second);
}
return ;
}
同时,因为pair的比较方法,这里必须把步数转换成负数的。如果不这样就必须写比较器:
bool operator < (const std::pair<int,int> &a,const std::pair<int,int> &b) {
if (a.first != b.first) return a.first < b.first;
return a.second > b.second;
}
bool operator <= (const std::pair<int,int> &a,const std::pair<int,int> &b) {
return a == b || a < b;
}
也可以用写cmp的方式,像sort一样传入第三个参数cmp,同时把dp[Q.back()] <= dp[i-g]改成(max(dp[Q.back()],dp[i-g],cmp) == dp[i-g]) 就行,但是不如上面那样写比较器来的逻辑清晰。
同时,这题有个奇怪的地方在于,重载了以后max还是会出错,必须 if (ret < dp[i]) ret = dp[i];来代替max才行。但是,自己试验了别的代码以后发现,重载了以后max函数是可以生效的,这里真是有毒啊。还是”留坑以后填“吧。。
zstu 4237 马里奥的求救——(单调队列DP)的更多相关文章
- POJ 3017 单调队列dp
Cut the Sequence Time Limit: 2000MS Memory Limit: 131072K Total Submissions: 8764 Accepted: 2576 ...
- [TyvjP1313] [NOIP2010初赛]烽火传递(单调队列 + DP)
传送门 就是个单调队列+DP嘛. ——代码 #include <cstdio> ; , t = , ans = ~( << ); int q[MAXN], a[MAXN], f ...
- 1304F2 - Animal Observation (hard version) 线段树or单调队列 +DP
1304F2 - Animal Observation (hard version) 线段树or单调队列 +DP 题意 用摄像机观察动物,有两个摄像机,一个可以放在奇数天,一个可以放在偶数天.摄像机在 ...
- vijos P1243 生产产品(单调队列+DP)
P1243生产产品 描述 在经过一段时间的经营后,dd_engi的OI商店不满足于从别的供货商那里购买产 品放上货架,而要开始自己生产产品了!产品的生产需要M个步骤,每一个步骤都可以在N台机器 ...
- POJ 1821 单调队列+dp
题目大意:有K个工人,有n个墙,现在要给墙涂色.然后每个工人坐在Si上,他能刷的最大范围是Li,且必须是一个连续子区间,而且必须过Si,他刷完后能获得Pi钱 思路:定义dp[i][j]表示前i个人,涂 ...
- codeforces 1077F2. Pictures with Kittens (hard version)单调队列+dp
被队友催着上(xun)分(lian),div3挑战一场蓝,大号给基佬紫了,结果从D开始他开始疯狂教我做人??表演如何AKdiv3???? 比赛场上:A 2 分钟,B题蜜汁乱计数,结果想得绕进去了20多 ...
- 【LOJ#10180】烽火传递 单调队列+dp
题目大意:给定一个 N 个非负整数数组成的序列,每个点有一个贡献值,现选出其中若干数,使得每连续的 K 个数中至少有一个数被选,要求选出的数贡献值最小. 题解:设 \(dp[i]\) 表示考虑了序列前 ...
- BZOJ 5281--[Usaco2018 Open]Talent Show(分数规划&单调队列&DP)
5281: [Usaco2018 Open]Talent Show Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 79 Solved: 58[Sub ...
- 【CF1077F2】Pictures with Kittens 单调队列+dp
题目大意:给定一个长度为 N 的序列,点有点权,从序列中选出恰好 X 个数,并且保证任意连续的 K 个数中均有一个被选中,求选出的点权最大是多少. 题解:此题可以作为 烽火传递+ 来处理,只不过在烽火 ...
随机推荐
- 怎样理解undefined和 null
前言: undefined表示 "未定义", null 表示 "空" 第一步: 一般在变量或属性没有声明或者声明以后没有赋值时, 这个变量的值就是undefin ...
- CentOS7利用systemctl添加dotnet后台服务
/usr/lib/systemd/system/zl.service systemctl enable zl.service systemctl start zl.service [Unit]Desc ...
- 初学java3 条件判断
三目运算符 条件? 正确结果:错误结果 if判断 单一条件判断 if(条件){ }else{ } 多种条件判断 if(){ }else if(){ } ... else{ } switch判断 swi ...
- Django中 auto_now_add 和 auto_now 的区别
auto_now_add = True #创建时添加的时间 修改数据时,不会发生改变 auto_now = True #修改数据的时间,每次修改都会有变动 ........
- Socket的神秘面纱
Tcp/IP协议是目前世界上使用最为广泛的协议,是以Tcp/IP为基础多个层次上的协议的集合.也称Tcp/IP协议族或Tcp/IP协议栈. TCP: Transmission Control Prot ...
- PS常识及技巧
常用格式 JPG:压缩 PNG:透明 GIF:动图 PSD:分层 分辨率 UI选择像素,印刷选择厘米 UI设计:72px 印刷分辨率必须为300 颜色模式UI网页设计:RGB 印刷类设计 ...
- 如何对Nginx日志文件进行切割保存
日积月累下,日志文件会越来越大,日志文件太大严重影响服务器效率,须要定时对日志文件进行切割. 切割的方式有按月切割.按天切割.按小时切割,一般都是按天切割. 那么如何进行切割呢? 思路: 创建日志文件 ...
- iOS13新坑(转自Cocoachina)
1.用presentViewController而非navigator,但在iOS13里默认是可下拉折叠的对话框,这样带来一个界面排版的高度并不是屏幕高度,从而影响界面效果.可以将viewcontro ...
- 处理器拦截器(HandlerInterceptor)详解(转)
简介 SpringWebMVC的处理器拦截器,类似于Servlet开发中的过滤器Filter,用于处理器进行预处理和后处理. 应用场景 1.日志记录,可以记录请求信息的日志,以便进行信息监控.信息统计 ...
- Linux文件的管理
创建:touch vim/vi echo重定向 cat touch 管理: atime mtime touch 文件名 //如果文件不存在,不创建文件 删除:mv /tmp find |x ...