NOIP 提高组 2014 飞扬的小鸟(记录结果再利用的DP)
https://www.cnblogs.com/violet-acmer/p/9937201.html
参考资料:
[1]:https://www.luogu.org/blog/xxzh2425/fei-yang-di-xiao-niao-ti-xie-p1941-post
[2]:https://www.luogu.org/blog/JOE/solution-p1941
需注意的地方:
(1):在每一时刻都可以点击屏幕好多好多次,就算是在m高度处也可以点击屏幕使其保持在最高点。
题解:
相关变量解释:
int n,m,k;
int x[maxn];//x[i] : 在X=i处点击屏幕,在i+1处上升x[i]高度
int y[maxn];//y[i] : 在X=i处不点击屏幕,在i+1处下降y[i]高度
struct Node
{
int id;//水管的编号
int l,h;//l : 下边界; h : 下边界
Node(int a=,int b=,int c=):id(a),l(b),h(c){}
}pipeline[maxn];//水管信息
int dp[maxn][*];//dp[i][j] : 来到i处的j高度所需的最少的点击量,至于为什么列要开2*1000,一会解释
根据dp定义,很容易写出状态转移方程:
dp[i][j]=min(dp[i][j],dp[i-1][j-k*x[i-1]]+k);
dp[i][j]=min(dp[i][j],dp[i-1][j+y[i-1]]);
1是指从(i-1,j-k*x[i-1])处点击屏幕k次来到(i,j)处
2是指从(i-1,j+y[i-1])处不点击屏幕来到(i,j)处,最终答案就是1,2中最小的那个
如果将此状态写出的代码提交上去,会超时的,为什么呢?
因为每个点(i,j)都需要循环 k 次来找出最小点击量,这就是O(Σni=1Σmj=1kj)的复杂度,而O(Σni=1Σmj=1kj)最大为O(n*m2)。
那要怎么办呢?注意观察一下:
假设来到(i,10)处,x[i-1]=3
在计算dp[ i ][10]的时候,dp[ i-1][4],dp[ i-1][1]相对大小已经在计算dp[i][7]的时候计算过了,所以应利用好之前的结果,那么状态转移方程就变为:
dp[i][j]=min(dp[i-1][j-x[i-1]]+1,dp[i][j-x[i-1]]+1);
dp[i][j]=min(dp[i][j],dp[i-1][j+y[i-1]]);
下面来证明一下正确性:
如果dp[i][7]=dp[i-1][4] => dp[i-1][4]+1 < dp[i-1][1]+2;
方程两端同时加上1 => dp[i-1][4]+2 < dp[i-1][1]+3,那来到 j = 10时,dp[i-1][4]+2与dp[i-1][1]+3的相对大小已经在求解dp[i][7]的时候求解出来了。
根据上述转移方程得到:
void updataDp(int i,int a,int b)
{
for(int j=+x[i-];j <= m+x[i-];++j)//从(i-1,j-x[i-1])处点击屏幕来到(i,j)处
dp[i][j]=min(dp[i-][j-x[i-]]+,dp[i][j-x[i-]]+); for(int j=;j < m;++j)//特判(i,m)点
{
int tot=(m-j)/x[i-];
while(j+tot*x[i-] < m)
tot++;
dp[i][m]=min(dp[i][m],dp[i-][j]+tot);
} for(int j=;j+y[i-] <= m;++j)//从(i-1,j+y[i-1])处不点击屏幕来到(i,j)处
dp[i][j]=min(dp[i][j],dp[i-][j+y[i-]]); dp[i][m]=min(dp[i][m],dp[i-][m]+);//就算在i-1处到达m点,也可以通过点击一次屏幕来到(i,m)处 for(int j=;j < a;++j)//[a,b]是i处无管道的区域,[a,b]之外都不可达,所以赋值为INF
dp[i][j]=INF;
for(int j=b+;j <= m;++j)
dp[i][j]=INF;
}
其中(i,m)点的特判可改为
for(int j=m+;j <= m+x[i-];++j)
dp[i][m]=min(dp[i][m],dp[i][j]);
这是为什么第一个for( )的范围最大到 m+x[i-1],以及dp[][]的列开到2*1000的原因;
AC代码:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
#define INF 0x3f3f3f3f
#define mem(a,b) memset(a,b,sizeof(a))
const int maxn=1e4+; int n,m,k;
int x[maxn];//x[i] : 在X=i处点击屏幕,在i+1处上升x[i]高度
int y[maxn];//y[i] : 在X=i处不点击屏幕,在i+1处下降y[i]高度
struct Node
{
int id;//水管的编号
int l,h;//l : 下边界; h : 下边界
Node(int a=,int b=,int c=):id(a),l(b),h(c){}
}pipeline[maxn];//水管信息
int dp[maxn][*];//dp[i][j] : 来到i处的j高度所需的最少的点击量,至于为什么列要开2*1000,一会解释
bool cmp(Node _a,Node _b){
return _a.id < _b.id;//按照管道编号升序排列
}
void Updata(int &a,int &b,int &ki,int i)//更新 X=i 处的上下边界
{
if(ki <= k && pipeline[ki].id == i)
a=pipeline[ki].l+,b=pipeline[ki].h-,ki++;
}
void updataDp(int i,int a,int b)
{
for(int j=+x[i-];j <= m+x[i-];++j)//从(i-1,j-x[i-1])处点击屏幕来到(i,j)处
dp[i][j]=min(dp[i-][j-x[i-]]+,dp[i][j-x[i-]]+); for(int j=;j < m;++j)//特判(i,m)点
{
int tot=(m-j)/x[i-];
while(j+tot*x[i-] < m)
tot++;
dp[i][m]=min(dp[i][m],dp[i-][j]+tot);
} for(int j=;j+y[i-] <= m;++j)//从(i-1,j+y[i-1])处不点击屏幕来到(i,j)处
dp[i][j]=min(dp[i][j],dp[i-][j+y[i-]]); dp[i][m]=min(dp[i][m],dp[i-][m]+);//就算在i-1处到达m点,也可以通过点击一次屏幕来到(i,m)处 for(int j=;j < a;++j)//[a,b]是i处无管道的区域,[a,b]之外都不可达,所以赋值为INF
dp[i][j]=INF;
for(int j=b+;j <= m;++j)
dp[i][j]=INF;
}
int Check()
{
int res=dp[][];
for(int i=;i <= m;++i)
res=min(dp[n][i],res);
return res;
}
int maxPass()
{
for(int ki=k;ki >= ;--ki)
for(int i=pipeline[ki].l+;i < pipeline[ki].h;++i)
if(dp[pipeline[ki].id][i] < INF)//为什么用 < 而不是用 != 呢?
return ki;
return ;
}
void Solve()
{
sort(pipeline+,pipeline+k+,cmp);
mem(dp,INF);
for(int i=;i <= m;++i)
dp[][i]=;
int ki=;
for(int i=;i <= n;++i)
{
int a=,b=m;
Updata(a,b,ki,i);//更新i处的无管道范围[a,b]
updataDp(i,a,b);
}
int res=Check();
if(res < INF)//为什么用 < 而不是用 != 呢?
printf("%d\n%d\n",,res);
else
printf("%d\n%d\n",,maxPass());
}
int main()
{
scanf("%d%d%d",&n,&m,&k);
for(int i=;i < n;++i)
scanf("%d%d",x+i,y+i);
for(int i=;i <= k;++i)
{
int a,b,c;
scanf("%d%d%d",&a,&b,&c);
pipeline[i]=Node(a,b,c);
}
Solve();
}
对代码中的问题解释一下,这是我下午踩的一个坑:
看updataDp中的第一个for()
for(int j=+x[i-];j <= m+x[i-];++j)
dp[i][j]=min(dp[i-][j-x[i-]]+,dp[i][j-x[i-]]+);
如果dp[i-1][j-x[i-1]] == INF 且 dp[i][j-x[i-1]] == INF,那dp[ i ][ j ] == INF+1 > INF;。
还发现一个有趣的地方:
mem(dp,0x3f) <=> mem(dp,0x3f3f3f3f)
但是 0x3f < 0x3f3f3f3f
NOIP 提高组 2014 飞扬的小鸟(记录结果再利用的DP)的更多相关文章
- 【NOIP2014提高组】飞扬的小鸟
https://www.luogu.org/problem/show?pid=1941 从某一点开始飞直到飞出地图最少点击屏幕的次数,显然只和该点的坐标唯一相关,没有后效性,考虑用DP解.令f(i,j ...
- NOIP 提高组 2014 联合权值(图论???)
传送门 https://www.cnblogs.com/violet-acmer/p/9937201.html 题解: 相关变量解释: int n; int fa[maxn];//fa[i] : i的 ...
- 题解【luoguP1351 NOIp提高组2014 联合权值】
题目链接 题意:给定一个无根树,每个点有一个权值.若两个点 \(i,j\) 之间距离为\(2\),则有联合权值 \(w_i \times w_j\).求所有的联合权值的和与最大值 分析: 暴力求,每个 ...
- poj 2229 Sumsets(记录结果再利用的DP)
传送门 https://www.cnblogs.com/violet-acmer/p/9852294.html 题意: 将一个数N分解为2的幂之和共有几种分法? 题解: 定义dp[ i ]为数 i 的 ...
- 津津的储蓄计划 NOIp提高组2004
这个题目当年困扰了我许久,现在来反思一下 本文为博客园ShyButHandsome的原创作品,转载请注明出处 右边有目录,方便快速浏览 题目描述 津津的零花钱一直都是自己管理.每个月的月初妈妈给津津\ ...
- NOIP提高组2004 合并果子题解
NOIP提高组2004 合并果子题解 描述:在一个果园里,多多已经将所有的果子打了下来,而且按果子的不同种类分成了不同的堆.多多决定把所有的果子合成一堆. 每一次合并,多多可以把两堆果子合并到一起,消 ...
- 计蒜客 NOIP 提高组模拟竞赛第一试 补记
计蒜客 NOIP 提高组模拟竞赛第一试 补记 A. 广场车神 题目大意: 一个\(n\times m(n,m\le2000)\)的网格,初始时位于左下角的\((1,1)\)处,终点在右上角的\((n, ...
- 1043 方格取数 2000 noip 提高组
1043 方格取数 2000 noip 提高组 题目描述 Description 设有N*N的方格图(N<=10,我们将其中的某些方格中填入正整数,而其他的方格中则放入数字0.如下图所示(见样 ...
- [NOIP提高组2018]货币系统
[TOC] 题目名称:货币系统 来源:2018年NOIP提高组 链接 博客链接 CSDN 洛谷博客 洛谷题解 题目链接 LibreOJ(2951) 洛谷(P5020) 大视野在线评测(1425) 题目 ...
随机推荐
- 思维导图,UML图,程序流程图制作从入门到精通
工具: https://www.processon.com/ 第一 用例图 第二 时序图 第三 流程图
- cuda编程-卷积优化
CUDA Convolution https://www.evl.uic.edu/sjames/cs525/final.html Improve Image Processing Using GPU ...
- 【C/C++】c文件重点总结
c文件重点知识总结 程序文件数据文件--->分文本文件(ASCII文件)和映像文件(二进制文件) .区分是用记事本打开后能否看懂. 用二进制文件读写花费时间少,因为用文本文件需要有一个转换的过程 ...
- JMeter——JMeter如何进行汉化
1.找到bin目录下的jmeter.properties文件 2.打开找到第37行,打开注释并将language=en改为language=zh_CN 3.重启
- Power Network POJ - 1459 网络流 DInic 模板
#include<cstring> #include<cstdio> #define FOR(i,f_start,f_end) for(int i=f_startl;i< ...
- 【BZOJ2144】Throw 数论
题目大意 给你三个数\(a,b,c\),每次你可以选择一个数\(s_1\),再选择一个数\(s_2\),把\(s_1\)变成\(2s_2-s_1\),但要求\(s_3\)不在\(s_1\)到\(2s_ ...
- 2018-01微信小程序--直播
一. 小程序直播支持的格式 目前小程序支付两种格式直播 1) flv格式直播 2) rtmp格式直播 二. 能够开通小程序直播的行业类目 由于直播需要资质, 并不是每个企业都能够开通小程序直播, 微信 ...
- maven编译时出现There are test failures
[ERROR] Failed to execute goal org.apache.maven.plugins:maven-surefire-plugin:2.10:test (default-tes ...
- 【 HDU 2177 】取(2堆)石子游戏 (威佐夫博弈)
BUPT2017 wintertraining(15) #5C hdu2177 题意 两个人轮流取石子,可以取一堆的任意非负整数个或两堆取相同个,先取完的输. 给定若干组数据:a,b表示两堆的石子数量 ...
- zabbix python 微信告警脚本
测试zabbix的微信告警耗费了大量时间,使用了开源工具(OneOaaS weixin-alert).shell脚本工具(手动执行正常,服务器调用失败),均没有实现相关功能以下是自己优化过的Pytho ...