解题报告(1)——飞扬的小鸟

2023-11-02 11:48
文章标签 报告 解题 飞扬 小鸟

本文主要是介绍解题报告(1)——飞扬的小鸟,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

飞扬的小鸟【NOIP2014提高组】

题目背景

NOIP2014提高组 Day1试题。

题目描述

Flappy Bird是一款风靡一时的休闲手机游戏。玩家需要不断控制点击手机屏幕的频率来调节小鸟的飞行高度,让小鸟顺利通过画面右方的管道缝隙。如果小鸟一不小心撞到了水管或者掉在地上的话,便宣告失败。


为了简化问题,我们对游戏规则进行了简化和改编:

1.游戏界面是一个长为 n,高为 m 的二维平面,其中有k个管道(忽略管道的宽度)。

2.小鸟始终在游戏界面内移动。小鸟从游戏界面最左边任意整数高度位置出发,到达游戏界面最右边时,游戏完成。

3.小鸟每个单位时间沿横坐标方向右移的距离为 1,竖直移动的距离由玩家控制。如果点击屏幕,小鸟就会上升一定高度 X,每个单位时间可以点击多次,效果叠加;如果不点击屏幕,小鸟就会下降一定高度 Y。小鸟位于横坐标方向不同位置时,上升的高度 X 和下降的高度 Y可能互不相同。

4.小鸟高度等于 0或者小鸟碰到管道时,游戏失败。小鸟高度为 m时,无法再上升。

 

现在,请你判断是否可以完成游戏。如果可以,输出最少点击屏幕数;否则,输出小鸟最多可以通过多少个管道缝隙。

 

输入格式

1行有 3个整数 nmk,分别表示游戏界面的长度,高度和水管的数量,每两个整数之间用一个空格隔开。

 

接下来的 n行,每行 2个用一个空格隔开的整数 X Y,依次表示在横坐标位置 0n-1上玩家点击屏幕后,小鸟在下一位置上升的高度 X,以及在这个位置上玩家不点击屏幕时,小鸟在下一位置下降的高度 Y

 

接下来 k行,每行 3个整数 PLH,每两个整数之间用一个空格隔开。每行表示一个管道,其中 P 表示管道的横坐标,L表示此管道缝隙的下边沿高度为 LH表示管道缝隙上边沿的高度(输入数据保证 P各不相同,但不保证按照大小顺序给出)。

 

输出格式

输出文件共两行:

第一行,包含一个整数,如果可以成功完成游戏,则输出 1,否则输出 0

第二行,包含一个整数,如果第一行为 1,则输出成功完成游戏需要最少点击屏幕数,否则,输出小鸟最多可以通过多少个管道缝隙。

 

样例数据 1

输入 

10 10 6

3 9

9 9

1 2

1 3

1 2

1 1

2 1

2 1

1 6

2 2

1 2 7

5 1 5

6 3 5

7 5 8

8 7 9

9 1 3

输出

1

6

样例数据 2

输入 

10 10 4

1 2

3 1

2 2

1 8

1 8

3 2

2 1

2 1

2 2

1 2

1 0 2

6 7 9

9 1 4

3 8 10

输出

0

3

备注

【样例说明】

如下图所示,蓝色直线表示小鸟的飞行轨迹,红色直线表示管道。

【数据范围】

对于 30%的数据:5n105m10k=0,保证存在一组最优解使得同一单位时间最多点击屏幕 3 次;

对于 50%的数据:5n205m10,保证存在一组最优解使得同一单位时间最多点击屏幕 3 次;

对于 70%的数据:5n10005m100

对于 100%的数据:5n100005m10000k<n0<X<m0<Y<m0<P<n0L<HmL+1<H

 

算法分析:

方法一:直接模拟+动归 预计得分:70

1)f[i][j]表示在横坐标为i,纵坐标为j时,最小的跳跃次数。(因担心MLE,所以实际代码由滚动数组处理),初始化f[0][j]全部为零。

2)依次枚举每个横纵坐标,,判断其是否在上轮可以到达,再依次将本轮可到达的位置标注记录。(注意不要超过上下限)

3)若有管道进行判断是否在可行范围内。

4)每轮纵坐标循环后,将未更新的点标注为-1(不可到达),方便下轮判断。

5)若检测到某轮结束全部点位为-1,输出0,计算已过管道并输出,结束程序。

6)最后一轮成功进行完毕,比较f[n][j],得到不等于-1,且最小的一个,输出1,输出最小值,结束程序。


Source 

#include#include#include#include#includeusing namespace std;   int n,m,k,po,j1,j2,mi,sum,pic; /*n:横坐标;m:纵坐标;k:管道数; po:临时位置记录; j1,j2:滚动数组推到用*/ /*mi:记录最小跳跃次数;sum:记录本轮无法到达的点位个数;pic:记录已通过管道数*/ int up[10001],down[10001]; /*up[i]:若i-1时点击屏幕,i时的上升高度*/ /*down[i]:若i-1时不点击屏幕,i时的下降高度*/ int pid[10001],piu[10001]; /*pid[i]:i处管道下方的最高点*/ /*piu[i]:i处管道上方的最低点*/ int f[2][10001]; /*滚动数组,记录当前行和上一行的最优解*/   inline void R(int &v)/*读入优化*/ {       v=0;       char c=0;       bool p=true;       while(c>'9'||c<'0')       {             if(c=='-')             {                   p=false;             }             c=cin.get();       }       while(c<='9'&&c>='0')       {             v=(v<<3)+(v<<1)+c-'0';             c=cin.get();       }       if(p==false)       {             v=-v;       } }   int main(void) {       ios::sync_with_stdio(false);       cin.tie(NULL);              /*cin解绑*/       R(n);       R(m);       R(k);       for(int i=1;i<=n;++i)       {             R(up[i]);             R(down[i]);       }       for(int i=1;i<=k;++i)       {             R(po);             R(pid[po]);             R(piu[po]);       }         for(int i=1;i<=m;++i)  /*初始化为0*/       {             f[0][i]=0;       }       j1=0;       j2=1;                  /*滚动数组的坐标初始化*/       for(int i=1;i<=n;++i)   /*枚举横坐标*/       {             if(piu[i]>0)        /*若当前列存在管道*/             {                   pic++;         /*管道计数加1*/             }             j1=(j1+1)%2;       /*改变滚动数组坐标*/             j2=(j2+1)%2;       /*改变滚动数组坐标*/             memset(f[j1],127,sizeof(f[j1])); /*初始化当前行最优解*/             for(int j=1;j<=m;++j)  /*枚举纵坐标*/             {                   if(f[j2][j]!=-1)    /*若上轮判定可以到达*/                   {                         int q=j;                         int cnt=0;                         while(q<=m)     /*判断点击次数和可以到达的位置*/                         {                               q=q+up[i];                               cnt++;     /*点击次数*/                               if(piu[i]!=0) /*若存在管道*/                               {                                     if(q pid[i]) /*保证在可以停留的范围*/                                     {                                           f[j1][q]=min(f[j2][j]+cnt,f[j1][q]);/*更新最优解*/                                     }                                     else                                     {                                           if(q>=piu[i]) /*若已经大于高处管道的最低点*/                                           {                                                 break;  /*跳出循环*/                                           }                                     }                               }                               else                               {                                     if(q>m)                                     {                                         f[j1][m]=min(f[j2][j]+cnt,f[j1][m]);                                         /*若超过高线,则停留在高线*/                                     }                                     else                                     {                                           f[j1][q]=min(f[j2][j]+cnt,f[j1][q]);                                           /*若未超过,更新当前位置*/                                                                        }                               }                         }                         q=j;                         cnt=0;     /*清零计数*/                         q=q-down[i]; /*判断若直接下降*/                         if(piu[i]!=0) /*若有管道*/                         {                               if(q pid[i])   /*判断是否满足条件*/                               {                                     f[j1][q]=min(f[j2][j],f[j1][q]); /*更新*/                               }                         }                         else                         {                               if(q>0)        /*若未到地面*/                               {                                     f[j1][q]=min(f[j2][j],f[j1][q]);/*更新*/                                                      }                         }                   }                  }       //      cout< <  

方法二:动归+优化 预计得分:100

1)f[i][j]表示在横坐标为i,纵坐标为j时,最小的跳跃次数。(因担心MLE,所以实际代码由滚动数组处理),初始化f[0][j]全部为零。

2)依次枚举每个纵坐标,,判断其是否在本轮可以到达,方法:判断j-up[i]本轮和上轮是否有有效值。(注意不要超过上下限)

3)单独判断m-up[i]m之间,本轮和上轮的可用值中的最小值。

4)最后将有管道的地方标注为不可行。

5)每轮纵坐标循环后,将未更新的点标注为-1(不可到达),方便下轮判断。

6)若检测到某轮结束全部点位为-1,输出0,计算已过管道并输出,结束程序。

7)最后一轮成功进行完毕,比较f[n][j],得到不等于-1,且最小的一个,输出1,输出最小值,结束程序。


Source 

Source :
#include#include#include#include#includeusing namespace std;   int n,m,k,po,j1,j2,mi,sum,pic; /*n:横坐标;m:纵坐标;k:管道数; po:临时位置记录; j1,j2:滚动数组推到用*/ /*mi:记录最小跳跃次数;sum:记录本轮无法到达的点位个数;pic:记录已通过管道数*/ int up[10001],down[10001]; /*up[i]:若i-1时点击屏幕,i时的上升高度*/ /*down[i]:若i-1时不点击屏幕,i时的下降高度*/ int pid[10001],piu[10001]; /*pid[i]:i处管道下方的最高点*/ /*piu[i]:i处管道上方的最低点*/ int f[2][10001]; /*滚动数组,记录当前行和上一行的最优解*/   inline void R(int &v)  /*读入优化*/ {       v=0;       char c=0;       bool p=true;       while(c>'9'||c<'0')       {             if(c=='-')             {                   p=false;             }             c=cin.get();       }       while(c<='9'&&c>='0')       {             v=(v<<3)+(v<<1)+c-'0';             c=cin.get();       }       if(p==false)       {             v=-v;       } }   int main(void) {       ios::sync_with_stdio(false);       cin.tie(NULL);   /*cin解绑*/       R(n);       R(m);       R(k);       for(int i=1;i<=n;++i)       {             R(up[i]);             R(down[i]);       }       for(int i=1;i<=k;++i)       {             R(po);             R(pid[po]);             R(piu[po]);       }         for(int i=1;i<=m;++i) /*初始化为0*/       {             f[0][i]=0;       }       j1=0;       j2=1;                /*滚动数组的坐标初始化*/       for(int i=1;i<=n;++i)   /*枚举横坐标*/       {             if(piu[i]>0)       /*若当前列存在管道*/             {                   pic++;         /*管道计数加1*/             }             j1=(j1+1)%2;       /*改变滚动数组坐标*/             j2=(j2+1)%2;       /*改变滚动数组坐标*/             memset(f[j1],127,sizeof(f[j1])); /*初始化当前行最优解*/             for(int j=1+up[i];j<=m;++j)  /*枚举当前列跳跃高度以上的点*/             {                   if(j!=m)                   {                         if(f[j2][j-up[i]]!=-1) /*判断上次起点能否达到*/                         {                               f[j1][j]=min(f[j1][j],f[j2][j-up[i]]+1); /*更新最优解*/                         }                         if(f[j1][j-up[i]]!=f[j1][0]) /*本轮之前位置可以到达*/                         {                               f[j1][j]=min(f[j1][j],f[j1][j-up[i]]+1); /*更新最优解*/                         }                   }                   else                   {                         for(k=m-up[i];k<=m;++k)  /*单独处理最顶点*/                         {                               if(f[j2][k]!=-1)    /*若上轮位置可用*/                               {                                     f[j1][m]=min(f[j2][k]+1,f[j1][m]); /*更新最优解*/                               }                               if(f[j1][k]!=f[j1][0])  /*本轮有位置可用*/                               {                                   f[j1][m]=min(f[j1][k]+1,f[j1][m]); /*更新最优解*/                               }                         }                   }             }             for(int j=1;j<=m-down[i];++j) /*枚举向下落点*/             {                   if(f[j2][j+down[i]]!=-1) /*若上轮位置可用*/                   {                         f[j1][j]=min(f[j1][j],f[j2][j+down[i]]); /*更新最优解*/                   }             }             if(piu[i]>0) /*最后判断管道*/             {                   for(int k=1;k<=pid[i];++k) /*处理下方管道*/                   {                         f[j1][k]=-1;                         sum++;                   }                   for(int k=piu[i];k<=m;++k) /*处理上方管道*/                   {                         f[j1][k]=-1;                         sum++;                   }             }                  for(int j=1;j<=m;++j) /*处理不能到达的位置*/           {                 if(f[j1][j]==f[j1][0])                 {                       f[j1][j]=-1;                       sum++; /*计数*/                 }           }             if(sum==m) /*全部不能到达*/           {                 cout<<"0"< 


Summary:

动态规划不像一般算法存在着使用的模板,每道题都有其不同的特点,只有认真揣摩分析每一道题的本质,再辅以所知道的算法,找到规划的方向才能够正确完成。当然,在考试中如果实在没有找到规划方向,就可以尝试用搜索或暴力模拟等极端方法,尽量得分。


这篇关于解题报告(1)——飞扬的小鸟的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/330393

相关文章

【专题】2024飞行汽车技术全景报告合集PDF分享(附原数据表)

原文链接: https://tecdat.cn/?p=37628 6月16日,小鹏汇天旅航者X2在北京大兴国际机场临空经济区完成首飞,这也是小鹏汇天的产品在京津冀地区进行的首次飞行。小鹏汇天方面还表示,公司准备量产,并计划今年四季度开启预售小鹏汇天分体式飞行汽车,探索分体式飞行汽车城际通勤。阅读原文,获取专题报告合集全文,解锁文末271份飞行汽车相关行业研究报告。 据悉,业内人士对飞行汽车行业

计算机毕业设计 大学志愿填报系统 Java+SpringBoot+Vue 前后端分离 文档报告 代码讲解 安装调试

🍊作者:计算机编程-吉哥 🍊简介:专业从事JavaWeb程序开发,微信小程序开发,定制化项目、 源码、代码讲解、文档撰写、ppt制作。做自己喜欢的事,生活就是快乐的。 🍊心愿:点赞 👍 收藏 ⭐评论 📝 🍅 文末获取源码联系 👇🏻 精彩专栏推荐订阅 👇🏻 不然下次找不到哟~Java毕业设计项目~热门选题推荐《1000套》 目录 1.技术选型 2.开发工具 3.功能

Python:豆瓣电影商业数据分析-爬取全数据【附带爬虫豆瓣,数据处理过程,数据分析,可视化,以及完整PPT报告】

**爬取豆瓣电影信息,分析近年电影行业的发展情况** 本文是完整的数据分析展现,代码有完整版,包含豆瓣电影爬取的具体方式【附带爬虫豆瓣,数据处理过程,数据分析,可视化,以及完整PPT报告】   最近MBA在学习《商业数据分析》,大实训作业给了数据要进行数据分析,所以先拿豆瓣电影练练手,网络上爬取豆瓣电影TOP250较多,但对于豆瓣电影全数据的爬取教程很少,所以我自己做一版。 目

开题报告中的研究方法设计:AI能帮你做什么?

AIPaperGPT,论文写作神器~ https://www.aipapergpt.com/ 大家都准备开题报告了吗?研究方法部分是不是已经让你头疼到抓狂? 别急,这可是大多数人都会遇到的难题!尤其是研究方法设计这一块,选定性还是定量,怎么搞才能符合老师的要求? 每次到这儿,头脑一片空白。 好消息是,现在AI工具火得一塌糊涂,比如ChatGPT,居然能帮你在研究方法这块儿上出点主意。是不

【干货分享】基于SSM的体育场管理系统的开题报告(附源码下载地址)

中秋送好礼 中秋佳节将至,祝福大家中秋快乐,阖家幸福。本期免费分享毕业设计作品:《基于SSM的体育场管理系统》。 基于SSM的体育场管理系统的开题报告 一、课题背景与意义 随着全民健身理念的深入人心,体育场已成为广大师生和社区居民进行体育锻炼的重要场所。然而,传统的体育场管理方式存在诸多问题,如资源分配不均、预约流程繁琐、数据统计不准确等,严重影响了体育场的使用效率和用户体验。

[SWPUCTF 2021 新生赛]web方向(一到六题) 解题思路,实操解析,解题软件使用,解题方法教程

题目来源 NSSCTF | 在线CTF平台因为热爱,所以长远!NSSCTF平台秉承着开放、自由、共享的精神,欢迎每一个CTFer使用。https://www.nssctf.cn/problem   [SWPUCTF 2021 新生赛]gift_F12 这个题目简单打开后是一个网页  我们一般按F12或者是右键查看源代码。接着我们点击ctrl+f后快速查找,根据题目给的格式我们搜索c

【中国国际航空-注册/登录安全分析报告】

前言 由于网站注册入口容易被黑客攻击,存在如下安全问题: 1. 暴力破解密码,造成用户信息泄露 2. 短信盗刷的安全问题,影响业务及导致用户投诉 3. 带来经济损失,尤其是后付费客户,风险巨大,造成亏损无底洞 所以大部分网站及App 都采取图形验证码或滑动验证码等交互解决方案, 但在机器学习能力提高的当下,连百度这样的大厂都遭受攻击导致点名批评, 图形验证及交互验证方式的安全性到底如

hdu1879(解题报告)

继续畅通工程                                   Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)

hdu2033(解题报告)

人见人爱A+B                                   Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)

HDU3791(解题报告)

二叉搜索树                      Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)                                          Total Subm