传送门

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

最新文章

  1. js的Object和Function
  2. Leetcode: Reconstruct Original Digits from English
  3. 《TCP/IP 详解 卷一》读书笔记 -----第四章 ARP
  4. MyBatis学习总结(一)
  5. K2十年:专注BPM
  6. codeforces 557 D. Vitaly and Cycle 组合数学 + 判断二分图
  7. php header 函数详解
  8. LeetCode题解——4Sum
  9. JS的第一节课
  10. 原子/Atomic操作
  11. SQL Server数据库空间管理 (1)
  12. 跨专业学习编程的苦逼生活 QWQ嘤嘤嘤
  13. Python下的OpenCV学习 02 —— 图像的读取与保存
  14. 201521123008&lt;java程序设计&gt;第三周实验总结
  15. iOS 中的特殊字面量表示方法
  16. Linux系统下yum镜像源环境部署记录
  17. python开发线程:线程&amp;守护线程&amp;全局解释器锁
  18. The markup in the document following the root element must be well-formed. Quartz.xml .......
  19. Set tooltip on customized tab header in WPF
  20. 2018-2019-2 《网络对抗技术》Exp3 免杀原理与实践 Week5 20165233

热门文章

  1. 在 Web 页面使用 VLC 插件播放 m3u8 视频流 (360 极速模式)
  2. OfficeToHtmlHelper
  3. JarvisOJ Basic 爱吃培根的出题人
  4. 数据库中事务的四大特性(ACID)
  5. python之旅六【第七篇】面向对象
  6. 洛谷P1119灾后重建
  7. Quartus prime 16.0 signaltap II 使用
  8. python3 函数function
  9. 【CF891C】Envy(最小生成树)
  10. JS分页条插件