传送门

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)的更多相关文章

  1. 【NOIP2014提高组】飞扬的小鸟

    https://www.luogu.org/problem/show?pid=1941 从某一点开始飞直到飞出地图最少点击屏幕的次数,显然只和该点的坐标唯一相关,没有后效性,考虑用DP解.令f(i,j ...

  2. NOIP 提高组 2014 联合权值(图论???)

    传送门 https://www.cnblogs.com/violet-acmer/p/9937201.html 题解: 相关变量解释: int n; int fa[maxn];//fa[i] : i的 ...

  3. 题解【luoguP1351 NOIp提高组2014 联合权值】

    题目链接 题意:给定一个无根树,每个点有一个权值.若两个点 \(i,j\) 之间距离为\(2\),则有联合权值 \(w_i \times w_j\).求所有的联合权值的和与最大值 分析: 暴力求,每个 ...

  4. poj 2229 Sumsets(记录结果再利用的DP)

    传送门 https://www.cnblogs.com/violet-acmer/p/9852294.html 题意: 将一个数N分解为2的幂之和共有几种分法? 题解: 定义dp[ i ]为数 i 的 ...

  5. 津津的储蓄计划 NOIp提高组2004

    这个题目当年困扰了我许久,现在来反思一下 本文为博客园ShyButHandsome的原创作品,转载请注明出处 右边有目录,方便快速浏览 题目描述 津津的零花钱一直都是自己管理.每个月的月初妈妈给津津\ ...

  6. NOIP提高组2004 合并果子题解

    NOIP提高组2004 合并果子题解 描述:在一个果园里,多多已经将所有的果子打了下来,而且按果子的不同种类分成了不同的堆.多多决定把所有的果子合成一堆. 每一次合并,多多可以把两堆果子合并到一起,消 ...

  7. 计蒜客 NOIP 提高组模拟竞赛第一试 补记

    计蒜客 NOIP 提高组模拟竞赛第一试 补记 A. 广场车神 题目大意: 一个\(n\times m(n,m\le2000)\)的网格,初始时位于左下角的\((1,1)\)处,终点在右上角的\((n, ...

  8. 1043 方格取数 2000 noip 提高组

    1043 方格取数  2000 noip 提高组 题目描述 Description 设有N*N的方格图(N<=10,我们将其中的某些方格中填入正整数,而其他的方格中则放入数字0.如下图所示(见样 ...

  9. [NOIP提高组2018]货币系统

    [TOC] 题目名称:货币系统 来源:2018年NOIP提高组 链接 博客链接 CSDN 洛谷博客 洛谷题解 题目链接 LibreOJ(2951) 洛谷(P5020) 大视野在线评测(1425) 题目 ...

随机推荐

  1. 设计模式笔记:单一职责原则(SRP, Single Responsibility Principle)

    1. 单一职责原则核心思想 一个类应该有且只有一个变化的原因. 2. 为什么引入单一职责原则 单一职责原则将不同的职责分离到单独的类,每一个职责都是一个变化的中心. 在SRP中,把职责定义为变化的原因 ...

  2. git 解决二进制文件冲突

    1.冲突的产生 当我们向远程git服务器提交某一个文件的修改时,恰巧这个文件相同的修改地方其他人也有修改,并且已经提交到服务器,这时冲突就产生了. 通常,当我们合并两个相同的地方都有修改的分支时,都会 ...

  3. 向继电器发送socket请求(python+java)

    近日,有一需求,向连接在内网的继电器发送socket请求,加以控制.原本并不复杂,只是io流/socket转换的问题,实操中却出现python代码没问题,java代码执行无响应的问题,问题很好定位:没 ...

  4. BZOJ3291Alice与能源计划——匈牙利算法+模拟费用流

    题目描述 在梦境中,Alice来到了火星.不知为何,转眼间Alice被任命为火星能源部长,并立刻面临着一个严峻的考验.为 了方便,我们可以将火星抽象成平面,并建立平面直角坐标系.火星上一共有N个居民点 ...

  5. BZOJ1014[JSOI2008]火星人——非旋转treap+二分答案+hash

    题目描述 火星人最近研究了一种操作:求一个字串两个后缀的公共前缀.比方说,有这样一个字符串:madamimadam,我们将这个字符串的各个字符予以标号:序号: 1 2 3 4 5 6 7 8 9 10 ...

  6. Sublime Text3 如何开启Debug

    打开setting-user 首选项——>Package Settings——>Package Control——>settings-user 添加"debug" ...

  7. 基于网络编程 TCP协议 及 socket 基本语法

    socket是什么 Socket是应用层与TCP/IP协议族通信的中间软件抽象层,它是一组接口.在设计模式中,Socket其实就是一个门面模式,它把复杂的TCP/IP协议族隐藏在Socket接口后面, ...

  8. windows下操作linux虚拟机映射网络驱动器中文件提示chmod权限不足解决方案

    为了方便操作,linux虚拟机会通过windows下连接网络驱动器的方式共享自己的文件,对于前端来说,我想把gulp放在windows磁盘,操作虚拟机中的php文件,一来节省虚拟机磁盘大小,二来解决虚 ...

  9. 【 HDU2966 】In case of failure(KD-Tree)

    BUPT2017 wintertraining(15) #5E HDU - 2966 题意 给平面直角坐标系下的n个点的坐标,求离每个点和它最近点的距离的平方.\(2 \le n \le 10^5\) ...

  10. python学习日记(深浅copy)

    赋值 #赋值,指向同一内存地址 l1 = [1,2,3,4,5] l2 = l1 print(l1,l2) print(id(l1),id(l2)) 浅copy #浅copy,第一层互相独立,创建了新 ...