被动态算法折磨的我看到了大神的这篇文章,觉得明白了许多,转载过来,以便回顾。然没询问大神意见,望谅解
大神使用C++写的,我这代码是java
原文地址为:https://blog.csdn.net/baidu_28312631/article/details/47418773

动态规划相信大家都知道,动态规划算法也是新手在刚接触算法设计时很苦恼的问题,有时候觉得难以理解,但是真正理解之后,就会觉得动态规划其实并没有想象中那么难。网上也有很多关于讲解动态规划的文章,大多都是叙述概念,讲解原理,让人觉得晦涩难懂,即使一时间看懂了,发现当自己做题的时候又会觉得无所适从。我觉得,理解算法最重要的还是在于练习,只有通过自己练习,才可以更快地提升。话不多说,接下来,下面我就通过一个例子来一步一步讲解动态规划是怎样使用的,只有知道怎样使用,才能更好地理解,而不是一味地对概念和原理进行反复琢磨。
首先,我们看一下这道题(此题目来源于北大POJ):
数字三角形(POJ1163)

在上面的数字三角形中寻找一条从顶部到底边的路径,使得路径上所经过的数字之和最大。路径上的每一步都只能往左下或 右下走。只需要求出这个最大和即可,不必给出具体路径。 三角形的行数大于1小于等于100,数字为 0 - 99
输入格式:
5 //表示三角形的行数 接下来输入三角形
7
3 8
8 1 0
2 7 4 4
4 5 2 6 5
要求输出最大和
接下来,我们来分析一下解题思路:
首先,肯定得用二维数组来存放数字三角形
然后我们用D( r, j) 来表示第r行第 j 个数字(r,j从1开始算)
我们用MaxSum(r, j)表示从D(r,j)到底边的各条路径中,最佳路径的数字之和。
因此,此题的最终问题就变成了求 MaxSum(1,1)
当我们看到这个题目的时候,首先想到的就是可以用简单的递归来解题:
D(r, j)出发,下一步只能走D(r+1,j)或者D(r+1, j+1)。故对于N行的三角形,我们可以写出如下的递归式

if(i==n-1) {Maxsun[i][j]=num[i][j];}else{Maxsun[i][j]=Math.max(MaxSum(i+1, j, num),MaxSum(i+1, j+1, num))+num[i][j];}

根据上面这个简单的递归式,我们就可以很轻松地写出完整的递归代码:

package other_demo;import java.util.Scanner;public class dp {static int n;public static int MaxSum(int i,int j,int[][] num) {if(i==n-1) {return num[i][j];}return Math.max(MaxSum(i+1, j, num),MaxSum(i+1, j+1, num))+num[i][j];}public static void main(String[] args) {Scanner sc=new Scanner(System.in);n=sc.nextInt();int[][] num=new int[n][n];for (int i = 0; i < n; i++) {for (int j = 0; j <= i; j++) {num[i][j]=sc.nextInt();}}System.out.println("最大路径为:"+MaxSum(0, 0,num));}
}

但是上面这个代码的运行时间会很长,因为我们重复计算了,当我们在进行递归时,计算机帮我们计算的过程如下图:

就拿第三行数字1来说,当我们计算从第2行的数字3开始的MaxSum时会计算出从1开始的MaxSum,当我们计算从第二行的数字8开始的MaxSum的时候又会计算一次从1开始的MaxSum,也就是说有重复计算。这样就浪费了大量的时间。也就是说如果采用递规的方法,深度遍历每条路径,存在大量重复计算。则时间复杂度为 2的n次方,对于 n = 100 行,肯定超时。

接下来,我们就要考虑如何进行改进,我们自然而然就可以想到如果每算出一个MaxSum(r,j)就保存起来,下次用到其值的时候直接取用,则可免去重复计算。那么可以用n方的时间复杂度完成计算。因为三角形的数字总数是 n(n+1)/2。
根据这个思路,我们就可以将上面的代码进行改进,使之成为记忆递归型的动态规划程序:

package other_demo;import java.util.Scanner;public class dp {static int n;public static int MaxSum(int i,int j,int[][] num,int[][] maxnum) {if(maxnum[i][j]!=-1) {return maxnum[i][j];}if(i==n-1) {return num[i][j];}maxnum[i][j]=Math.max(MaxSum(i+1, j, num,maxnum),MaxSum(i+1, j+1, num,maxnum))+num[i][j];return maxnum[i][j];}public static void main(String[] args) {Scanner sc=new Scanner(System.in);n=sc.nextInt();int[][] num=new int[n][n];//使用maxsum[i][j]来记录num[i][j]路径一下的最大路径值。int[][] maxnum=new int[n][n];for (int i = 0; i < n; i++) {for (int j = 0; j <= i; j++) {num[i][j]=sc.nextInt();maxnum[i][j]=-1;}}System.out.println("最大路径为:"+MaxSum(0, 0,num,maxnum));}}

虽然在短时间内就AC了。但是,我们并不能满足于这样的代码,因为递归总是需要使用大量堆栈上的空间,很容易造成栈溢出,我们现在就要考虑如何把递归转换为递推,让我们一步一步来完成这个过程。

我们首先需要计算的是最后一行,因此可以把最后一行直接写出,如下图:
现在开始分析倒数第二行的每一个数,现分析数字2,2可以和最后一行4相加,也可以和最后一行的5相加,但是很显然和5相加要更大一点,结果为7,我们此时就可以将7保存起来,然后分析数字7,7可以和最后一行的5相加,也可以和最后一行的2相加,很显然和5相加更大,结果为12,因此我们将12保存起来。以此类推。。我们可以得到下面这张图:

然后按同样的道理分析倒数第三行和倒数第四行,最后分析第一行,我们可以依次得到如下结果:


上面的推导过程相信大家不难理解,理解之后我们就可以写出如下的递推型动态规划程序:

package other_demo;import java.util.Scanner;public class dp {    public static void main(String[] args) {Scanner sc=new Scanner(System.in);int n;n=sc.nextInt();int[][] num=new int[n][n];int[][] maxnum=new int[n][n];for (int i = 0; i < n; i++) {for (int j = 0; j <= i; j++) {num[i][j]=sc.nextInt();}}for (int j =0; j <n; j++) {maxnum[n-1][j]=num[n-1][j];}for (int i = n-2; i >=0; i--) {for (int j = 0; j <= i; j++) {maxnum[i][j]=Math.max(maxnum[i+1][j], maxnum[i+1][j+1])+num[i][j];}}System.out.println("最大路径为:"+maxnum[0][0]);}
}

我们的代码仅仅是这样就够了吗?当然不是,我们仍然可以继续优化,而这个优化当然是对于空间进行优化,其实完全没必要用二维maxSum数组存储每一个MaxSum(r,j),只要从底层一行行向上递推,那么只要一维数组maxSum[100]即可,即只要存储一行的MaxSum值就可以。
对于空间优化后的具体递推过程如下:






接下里的步骤就按上图的过程一步一步推导就可以了。进一步考虑,我们甚至可以连maxSum数组都可以不要,直接用D的第n行直接替代maxSum即可。但是这里需要强调的是:虽然节省空间,但是时间复杂度还是不变的。
依照上面的方式,我们可以写出如下代码:

package other_demo;import java.util.Scanner;public class dp {    public static void main(String[] args) {Scanner sc=new Scanner(System.in);int n;n=sc.nextInt();int[][] num=new int[n][n];int[][] maxnum=new int[n][n];for (int i = 0; i < n; i++) {for (int j = 0; j <= i; j++) {num[i][j]=sc.nextInt();}}for (int j =0; j <n; j++) {maxnum[n-1][j]=num[n-1][j];}for (int i = n-2; i >=0; i--) {for (int j = 0; j <= i; j++) {maxnum[n-1][j]=Math.max(maxnum[n-1][j], maxnum[n-1][j+1])+num[i][j];}}System.out.println("最大路径为:"+maxnum[n-1][0]);}
}

作者:ChrisYoung1314
来源:CSDN
原文:https://blog.csdn.net/baidu_28312631/article/details/47418773
版权声明:本文为博主原创文章,转载请附上博文链接!

动态规划——入门(1)相关推荐

  1. LQ训练营(C++)学习笔记_动态规划入门

    动态规划入门 五.动态规划入门 1.动态介绍 1.1动态规划基本思路 1.2 动态规划基本概念 1.2.1 阶段 1.2.2 状态 1.2.3 决策 1.2.4 状态转移方程 1.2.5 策略 1.3 ...

  2. 动态规划入门之国王的金矿

    最近学习算法,对动态规划不太了解,使用的时候照搬转移方程式,知其然不知其所以然,今天看到一篇动态规划的教程,解释得非常通俗,原文在这里[动态规划入门教程] (http://blog.csdn.net/ ...

  3. 很特别的一个动态规划入门教程

    很特别的一个动态规划入门教程 今天在网上看到一个讲动态规划的文章,是以01背包为例的,这文章和书上的讲解非常不一样,令我眼前一亮,于是转载一下下--- (说明一下,本人非常痛恨教材公式定理漫天飞,实际 ...

  4. C++动态规划入门习题+解析

    动态规划入门 ❤️‍

  5. 【LeetCode】动态规划入门(专项打卡21天合集)

    [LeetCode]动态规划入门(专项打卡21天合集) 下图为证 文章目录 [LeetCode]动态规划入门(专项打卡21天合集) Day1 斐波拉契数 第 N 个泰波那契数 Day2 爬楼梯 使用最 ...

  6. 动态规划入门问题:猴子与香蕉

    动态规划入门问题猴子与香蕉 一.猴子与香蕉 题目描述 一组研究人员正在设计一个测试猴子IQ的实验.他们把香蕉吊在屋顶上,同时给猴子提供了砖块.如果猴子够聪明,它会把砖块一个个叠起来做成一个塔,然后爬上 ...

  7. 动态规划入门看这篇就够了,万字长文!

    今天是小浩算法 "365刷题计划" 动态规划 - 整合篇.大家应该期待已久了吧!奥利给! 01 PART 动态规划是啥 我们把要解决的一个大问题转换成若干个规模较小的同类型问题,当 ...

  8. 1075: 动态规划入门(中链式2:能量项链)

    1075: 动态规划入门(中链式2:能量项链) 时间限制: 1 Sec 内存限制: 128 MB 题目描述 [问题描述] 能量球组成的项链.相邻两球可以合并产生新球.合并规则:如果前一颗能量珠的头标记 ...

  9. 动态规划入门到熟悉,看不懂来打我啊

    动态规划入门到熟悉,看不懂来打我啊 兔子hebtu666 本文链接:https://blog.csdn.net/hebtu666/article/details/100585136 2.1斐波那契系列 ...

  10. 【精品计划1】动态规划入门到熟悉,看不懂来打我啊

    持续更新...... 2.1斐波那契系列问题 2.2矩阵系列问题 2.3跳跃系列问题 3.1 01背包 3.2 完全背包 3.3多重背包 3.4 一些变形选讲 2.1斐波那契系列问题 在数学上,斐波纳 ...

最新文章

  1. 搜狗分身技术再进化,让AI合成主播“动”起来
  2. 工程师必知的代码重构指南
  3. 推箱子java下载_Java实现简单推箱子游戏
  4. Why is it recommended to create clusters with odd number of nodes? | 为什么集群节点建议奇数个?
  5. MySQL Sandbox---快速体验各版本MySQL
  6. 欧司朗台灯的灯泡容易坏是怎么回事
  7. 前后端分离的思考与实践(二)
  8. 卷积神经网络中十大令人拍案叫绝的操作
  9. 计算机access预测题,2016计算机二级《Access》考试预测题及答案
  10. 【干货】如何打造一流创业团队-创业者最实用的管理指南.pdf(附下载链接)...
  11. 旧项目归档:旅游年卡-旅游直通车-微信二级分销推广会员-汽车租赁-金币提现-司机乘务管理-景点线路乘车预约
  12. 手把手教你给女朋友编写一个公众号定时推送(java版本)
  13. lsnrctl command not found
  14. php 九宫格验证码,PHP九宫格抽奖源码示例
  15. 研发内部控制浅谈(一)(转)
  16. 斯坦福NLP课程 | 第11讲 - NLP中的卷积神经网络
  17. 美国商务部机构建议这样生成软件供应链 “身份证”
  18. 我很忙!你忙什么?瞎忙!
  19. 统计学习笔记:方差分析
  20. MPLS-VPN实验

热门文章

  1. 如何解释混合网络?—Vecloud微云
  2. 发现一个电子书下载的【简书】
  3. P4391 [BOI2009]Radio Transmission 无线传输
  4. 《Java大学教程》—第5章 数组
  5. Java Web开发中路径问题小结
  6. KingDZ 变菜鸟,每日一个C#小实例之---百叶窗图片展示
  7. 将在2021年改变商业格局的10项技术
  8. 数据从业者必读:抓取了一千亿个网页后我才明白,爬虫一点都不简单
  9. 计算机视觉库OpenCV之saturate_cast防数据越界函数
  10. AI专家Marcus质疑深度学习:面临十大挑战(含参考文献)