热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

动态规划入门之小明课间爬台阶

从前有一个很可爱的小学生叫做小明,小明平时最大的爱好就是思考问题。有一天,小明走到了教学楼的台阶前,小明心想:“来学校上学都有好几天了,竟然没有数过每一层有多少台阶”,然后他就开始数了起来:

从前有一个很可爱的小学生叫做小明,小明平时最大的爱好就是思考问题。
有一天,小明走到了教学楼的台阶前,小明心想:“来学校上学都有好几天了,竟然没有数过每一层有多少台阶”,然后他就开始数了起来:
1、2、……
“一共有10个台阶!”
数完了台阶,小明开开心心地上台阶了。可是小明边上台阶,他又边思考呀:“我每次跨一步可以上一级台阶,也可以上二级台阶,再用力跨的话还可以上三级台阶。那么我从最底下的地面走到最上层的10级台阶,有多少方案数呢?”
这个问题引起了小明的思考,但是小明是一个勤于思考的好学生,他绝对不会让一个问题难倒他的。
小明于是开始对这个问题进行全面的思考:
假设我现在站在台阶前,我所在的地面是第0层台阶,我可以到达的台阶分别是第1、2、3、……、10层台阶,那么这个问题就变成了:
“请问从第0层台阶开始,每次可以走1、2或3层台阶,最终到达第10层台阶的方案数有多少种?”
然后小明开始思考:
从第0层台阶走到第0层台阶的方案数一共有1种(因为小明现在就站在第0层台阶上,所以它不需要懂就能到达第0层台阶,也就是说只有1种方案)。
从第0层台阶走到第1层台阶的方案数一共有1种:
    0 --> 1 (表示从第0层台阶走到第1层台阶,下同)
从第0层台阶走到第2层台阶的方案数一共有2种:
    0 --> 2 (表示直接从第0层台阶跨到了第2层台阶)
    0 --> 1 --> 2 (表示先从第0层台阶跨到了第1层台阶,再从第1层台阶跨到了第2层台阶)
从第0层台阶走到第3层台阶的方案数一共有4种:
    0 --> 3
    0 --> 2 --> 3
    0 --> 1 --> 3
    0 --> 1 --> 2 --> 3
从第0层台阶走到第4层台阶的方案数一共有7种:
    0 --> 3 --> 4
    0 --> 2 --> 3 --> 4
    0 --> 1 --> 3 --> 4
    0 --> 1 --> 2 --> 3 --> 4
    0 --> 2 --> 4
    0 --> 1 --> 2 --> 4
    0 --> 1 --> 4
……
小明非常仔细地列举出了从第0层台阶到第0、1、2、3、4层台阶的所有方案数,但是当小明开始列举从第0层台阶到第5层台阶的时候他戛然而止。“不行,这不是我理想种解决问题的方案。如果按照这种方案算到第10层,不仅容易漏掉方案导致算错,还会花费大量时间,在课间十分钟的时间内我是绝对算不完的!”
于此同时,小明也发现了方案数之间的规律。
假设F(i)表示小明从第0层台阶走到第i层台阶的方案数,那么,可以得出下面的规律:
    F(0) = 1
    F(1) = F(0) = 1
    F(2) = F(1) + F(0) = 1 + 1 = 2
而对于所有满足3<=n<=10的n来说,可以得到:
    F(n) = F(n-1) + F(n-2) + F(n-3)
于是,我们可以按照这个公式来得出解决方案如下:
    F(3) = F(2) + F(1) + F(0) = 2 + 1 + 1 = 4
    F(4) = F(3) + F(2) + F(1) = 4 + 2 + 1 = 7
    F(5) = F(4) + F(3) + F(2) = 7 + 4 + 2 = 13
    F(6) = F(5) + F(4) + F(3) = 13 + 7 + 4 = 24
    F(7) = F(6) + F(5) + F(4) = 24 + 13 + 7 = 44
    F(8) = F(7) + F(6) + F(5) = 44 + 24 + 13 = 81
    F(9) = F(8) + F(7) + F(6) = 81 + 44 + 24 = 149
    F(10) = F(9) + F(8) + F(7) = 149 + 81 + 44 = 274
于是,小明在上课预备铃想的那一刻,计算出了从第0层台阶到第10层台阶的方案数是274种,开开心心地去上课了。

趁小明去上课的时间,在这里和大家简要地阐述一下小明解决爬楼梯问题地算法,那就是“动态规划”了。
拿爬楼梯地问题来说,我们把到达某一层楼梯都设为一种状态,用F(i)来表示从第0层台阶爬到第i层台阶地方案数,那么我们可以发现,F(i)和F(i-1)、F(i-2)以及F(i-3)都是有联系的,进一步推敲他们之间的关系,我们便可以得出下面的公式:
    F(n) = F(n-1) + F(n-2) + F(n-3)
像这样的公式我们一般把它叫做“状态转移方程”,而像这种一般可以用状态转移方程进行求解问题的方法我们就把它们称为“动态规划”了。
现在,你是不是对动态规划有了一个大致的了解了呢:)

求解上述问题的C++代码:

#include 
int f[11];
int main()
{
f[
0] = f[1] = 1;
f[
2] = f[1] + f[0];
for (int i = 3; i <= 10; i ++)
{
f[i]
= f[i-1] + f[i-2] + f[i-3];
}
printf(
"%d\n", f[10]);
return 0;
}

 


推荐阅读
  • 本文为Codeforces 1294A题目的解析,主要讨论了Collecting Coins整除+不整除问题。文章详细介绍了题目的背景和要求,并给出了解题思路和代码实现。同时提供了在线测评地址和相关参考链接。 ... [详细]
  • HDU 2372 El Dorado(DP)的最长上升子序列长度求解方法
    本文介绍了解决HDU 2372 El Dorado问题的一种动态规划方法,通过循环k的方式求解最长上升子序列的长度。具体实现过程包括初始化dp数组、读取数列、计算最长上升子序列长度等步骤。 ... [详细]
  • 本文介绍了九度OnlineJudge中的1002题目“Grading”的解决方法。该题目要求设计一个公平的评分过程,将每个考题分配给3个独立的专家,如果他们的评分不一致,则需要请一位裁判做出最终决定。文章详细描述了评分规则,并给出了解决该问题的程序。 ... [详细]
  • 本文介绍了C++中省略号类型和参数个数不确定函数参数的使用方法,并提供了一个范例。通过宏定义的方式,可以方便地处理不定参数的情况。文章中给出了具体的代码实现,并对代码进行了解释和说明。这对于需要处理不定参数的情况的程序员来说,是一个很有用的参考资料。 ... [详细]
  • 本文主要解析了Open judge C16H问题中涉及到的Magical Balls的快速幂和逆元算法,并给出了问题的解析和解决方法。详细介绍了问题的背景和规则,并给出了相应的算法解析和实现步骤。通过本文的解析,读者可以更好地理解和解决Open judge C16H问题中的Magical Balls部分。 ... [详细]
  • 本文讨论了使用差分约束系统求解House Man跳跃问题的思路与方法。给定一组不同高度,要求从最低点跳跃到最高点,每次跳跃的距离不超过D,并且不能改变给定的顺序。通过建立差分约束系统,将问题转化为图的建立和查询距离的问题。文章详细介绍了建立约束条件的方法,并使用SPFA算法判环并输出结果。同时还讨论了建边方向和跳跃顺序的关系。 ... [详细]
  • c语言\n不换行,c语言printf不换行
    本文目录一览:1、C语言不换行输入2、c语言的 ... [详细]
  • 本文介绍了一种划分和计数油田地块的方法。根据给定的条件,通过遍历和DFS算法,将符合条件的地块标记为不符合条件的地块,并进行计数。同时,还介绍了如何判断点是否在给定范围内的方法。 ... [详细]
  • 本文介绍了P1651题目的描述和要求,以及计算能搭建的塔的最大高度的方法。通过动态规划和状压技术,将问题转化为求解差值的问题,并定义了相应的状态。最终得出了计算最大高度的解法。 ... [详细]
  • 本文介绍了解决二叉树层序创建问题的方法。通过使用队列结构体和二叉树结构体,实现了入队和出队操作,并提供了判断队列是否为空的函数。详细介绍了解决该问题的步骤和流程。 ... [详细]
  • 本文介绍了C函数ispunct()的用法及示例代码。ispunct()函数用于检查传递的字符是否是标点符号,如果是标点符号则返回非零值,否则返回零。示例代码演示了如何使用ispunct()函数来判断字符是否为标点符号。 ... [详细]
  • 动态规划算法的基本步骤及最长递增子序列问题详解
    本文详细介绍了动态规划算法的基本步骤,包括划分阶段、选择状态、决策和状态转移方程,并以最长递增子序列问题为例进行了详细解析。动态规划算法的有效性依赖于问题本身所具有的最优子结构性质和子问题重叠性质。通过将子问题的解保存在一个表中,在以后尽可能多地利用这些子问题的解,从而提高算法的效率。 ... [详细]
  • Linux环境变量函数getenv、putenv、setenv和unsetenv详解
    本文详细解释了Linux中的环境变量函数getenv、putenv、setenv和unsetenv的用法和功能。通过使用这些函数,可以获取、设置和删除环境变量的值。同时给出了相应的函数原型、参数说明和返回值。通过示例代码演示了如何使用getenv函数获取环境变量的值,并打印出来。 ... [详细]
  • 本文介绍了PE文件结构中的导出表的解析方法,包括获取区段头表、遍历查找所在的区段等步骤。通过该方法可以准确地解析PE文件中的导出表信息。 ... [详细]
  • C++中的三角函数计算及其应用
    本文介绍了C++中的三角函数的计算方法和应用,包括计算余弦、正弦、正切值以及反三角函数求对应的弧度制角度的示例代码。代码中使用了C++的数学库和命名空间,通过赋值和输出语句实现了三角函数的计算和结果显示。通过学习本文,读者可以了解到C++中三角函数的基本用法和应用场景。 ... [详细]
author-avatar
手机用户2602889207
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有