一、基本概念

   把多阶段过程转化为一系列单阶段问题,利用各阶段之间的关系,逐个求解。

    动态规划过程:每次决策依赖于当前状态,又随即引起状态的转移。一个决策序列就是在变化的状态中产生出来的,所以,这种多阶段最优化决策解决问题的过程就称为动态规划。
    假设问题是由交叠的子问题所构成,我们就能够用动态规划技术来解决它。一般来说,这种子问题出自对给定问题求解的递推关系中,这个递推关系包括了同样问题的更小子问题的解。动态规划法建议,与其对交叠子问题一次重新的求解,不如把每一个较小子问题仅仅求解一次并把结果记录在表中(动态规划也是空间换时间的)。这样就能够从表中得到原始问题的解。

二、基本思想与策略

    基本思想与分治法类似,也是将待求解的问题分解为若干个子问题(阶段),按顺序求解子阶段,前一子问题的解,为后一子问题的求解提供了实用的信息。

   在求解任一子问题时,列出各种可能的局部解,通过决策保留那些有可能达到最优的局部解,丢弃其它局部解。依次解决各子问题,最后一个子问题就是初始问题的解。

   因为动态规划解决的问题多数有重叠子问题这个特点。为降低反复计算。对每个子问题仅仅解一次,将其不同阶段的不同状态保存在一个二维数组中。

   与分治法最大的区别是:适合于用动态规划法求解的问题,经分解后得到的子问题往往不是互相独立的(即下一个子阶段的求解是建立在上一个子阶段的解的基础上,进行进一步的求解)。

三、适用的情况

能采用动态规划求解的问题的一般要具有3个性质:

(1)最优化原理:假设问题的最优解所包括的子问题的解也是最优的,就称该问题具有最优子结构,即满足最优化原理。

(2)无后效性:即某阶段状态一旦确定。就不受这个状态以后决策的影响。也就是说,某状态以后的过程不会影响曾经的状态。仅仅与当前状态有关;

(3)有重叠子问题:即子问题之间是不独立的,一个子问题在下一阶段决策中可能被多次使用到(该性质并非动态规划适用的必要条件,可是假设没有这条性质,动态规划算法同其它算法相比就不具备优势)。

四、求解的基本步骤

    动态规划所处理的问题是一个多阶段决策问题,一般由初始状态开始,通过对中间阶段决策的选择,达到结束状态。这些决策形成了一个决策序列,同时确定了完成整个过程的一条活动路线(通常是求最优的活动路线)。

  • 初始状态→│决策1│→│决策2│→…→│决策n│→结束状态

(1)划分阶段:按照问题的时间或空间特征,把问题分为若干个阶段。在划分阶段时,注意划分后的阶段一定要是有序的或者是可排序的,否则问题就无法求解

(2)确定状态和状态变量:将问题发展到各个阶段时所处于的各种客观情况用不同的状态表示出来。当然,状态的选择要满足无后效性。

(3)确定决策并写出状态转移方程:状态转移就是根据上一阶段的状态和决策来导出本阶段的状态。如果确定了决策,状态转移方程也就可写出。但事实上常常是反过来做,根据相邻两个阶段的状态之间的关系来确定决策方法和状态转移方程。

(4)寻找边界条件:给出的状态转移方程是一个递推式,需要一个递推的终止条件或边界条件。

只要解决问题的阶段、状态和状态转移决策确定了,就可以写出状态转移方程(包括边界条件)。

  • 确定状态:最后一步和转化子问题
  • 确定转移方程
  • 确定边界情况和初始条件
  • 确定好计算顺序

五、常见动态规划问题

1、找零钱问题(求最值动态规划)

有面值为1元、2元和7元的硬币若干枚,如何用最少的硬币凑够27元?

  • 1.确定状态

    • 最后一步:(最优策略中使用的最后一枚硬币ak)
    • 转化子问题:最少的硬币拼出更小的面值27-ak
  • 2.转移方程
    f[X] = min{f[X-2]+1 , f[X-5]+1 , f[x-7]+1}
  • 3.初始条件和边界情况
    f[0] = 0,如果不能拼出Y,f[Y] = 正无穷
  • 4.计算顺序
    f[0],f[1],f[2], …
  • 代码实现
class Solution:def coinChange(self, coins: List[int], amount: int) -> int:# 开数组 27 个位置的数组f = [float("inf")] * (amount+1) # 初始条件:0元0种方法f[0] = 0n = len(coins)#f[1],f[2]....f[27]#每个数组空间装当前零钱最优的拼凑结果for i in range(1,amount+1):#对每种零钱进行遍历#转移方程f[X] = min{f[X-2]+1 , f[X-5]+1 , f[x-7]+1}for j in range(0,n):if(i>=coins[j]):f[i] = min(f[i-coins[j]]+1,f[i])#如果凑不出目标数,返回-1if f[-1] == float("inf"):return -1#返回次数else:return f[-1]

2、走方格问题(计数型动态规划)

给定m行和n列网格,有一个机器人从左上角(0,0)出发,每一步可以向下或向右走一步,问多少种不同的方式走到右下角

1.确定状态

  • 最后一步:
    无论机器人用何种方式到达右下角,总有最后挪动的一步-向右或者向下
    右下角设为(m-1,n-1),机器人在前一步一定是(m-2,n-2)
  • 子问题:
    机器人有X种方式从左上角走到(m-2,n-1),Y种方式从左上角走到(m-1,n-2),则一共有X+Y种方式走到(m-1,n-1),一次类推
    两个变量,二维数组:设f[i][j]为机器人有多少种方式从左上角走到(i , j)

2.转移方程:对任意一个格子:f[i][j] = f[i-1][j]+f[i][j-1]

3.初始条件和边界情况:

  • 初始条件:f[0][0] = 1,只有一种方法到左上角
  • 边界情况:i = 0,j=0的时候,则前一步只有一个方向过来f[i][]j] = 1

4.计算顺序

求f[m-1][n-1](时间复杂度:O(MN) 空间复杂度(数组大小):O(MN))

5.代码实现

import numpy as np
def uniquePaths(m,n):   #m:行 ,n:列 5 4#如果用([0]*m)*n会涉及到拷贝问题f = np.zeros((m,n),dtype=np.int)for j in range(n):for i in range(m):#边界为0的情况下均为1步if(i==0 or j==0):f[i][j] = 1#转移方程else:f[i][j] = f[i-1][j]+f[i][j-1]return f[i][j]
uniquePaths(5,4)

3、青蛙过桥(存在型动态规划)

有n块石头分别在x轴的0,1,2,… ,n-1位置,一只青蛙在石头0处,如果青蛙在第i块石头上,它最多往右跳ai个石头。
问青蛙是否能够调到石头n-1处

1.确定状态

  • 最后一步:青蛙能到达石头i;石头i到石头n-1的距离(即最后一步的距离)不能超过可跳跃的最大距离即:n-1-i<=a[i]
  • 子问题:这样原问题青蛙能否跳到石头n-1转化为规模更小(i<n-1)的子问题青蛙能不能跳到石头i。
    状态:dp[j] = 青蛙能不能跳到石头j

2.转移方程:
设状态为dp[j] = 青蛙能不能跳到石头j,有:

3.初始条件和边界情况
初始条件:dp[0] = True,青蛙一开始就在石头0,无边界情况。

4.计算顺序
由初始条件dp[0] = True,
计算dp[1],dp[2],…,dp[n-1],结果:dp[n-1]
时间复杂度:O(n^2)(用贪心算法复杂度为O(n)),空间复杂度:O(n)

5.代码实现

五大常用算法—动态规划详解和经典题目(python)相关推荐

  1. C++ STL【常用算法】详解

    大家好,欢迎来到STL[常用算法]详解模块,本文将对STL中常见但又高频的算法进行一个详细又简单的入门级教学

  2. 五大常用算法——动态规划算法详解及经典例题

    一.基本概念 动态规划是运筹学中用于求解决策过程中的最优化数学方法.当然,我们在这里关注的是作为一种算法设计技术,作为一种使用多阶段决策过程最优的通用方法. 动态规划过程是:每次决策依赖于当前状态,又 ...

  3. python动态规划详解_经典动态规划例题整理(Python版)

    由于本人的算法基础较为薄弱,所以在这里整理一下自己的做过的题,使自己能够随时随地回顾温习. 然后,本篇文章将会持续更新自己遇到的一些比较经典动态规划的题目,大家如果对代码有任何问题,直接在文章下面评论 ...

  4. 常用算法 之三 详解 SHA1 实现(基于算法的官方原文档)及源码详细注释

    写在前面   在之前的工作中,用到了CRC16.MD5 和 SHA1 算法,主要用来校验下发的文件.网上关于这些算法的文章铺天盖地,以下内容仅仅是自己在学习时候的一个记录,一些套话来自于互联网.下面先 ...

  5. 常用算法 之一 详解 MD5 实现(基于算法的官方原文档)及源码详细注释

    写在前面   在之前的工作中,用到了CRC16.MD5 和 SHA1 算法,主要用来校验下发的文件.网上关于这些算法的文章铺天盖地,以下内容仅仅是自己在学习时候的一个记录,一些套话来自于互联网.下面先 ...

  6. (转)五大常用算法:分治、动态规划、贪心、回溯和分支界定

    分治算法 一.基本概念 在计算机科学中,分治法是一种很重要的算法.字面上的解释是"分而治之",就是把一个复杂的问题分成两个或更多的相同或相似的子问题,再把子问题分成更小的子问题-- ...

  7. 五大常用算法:分治、动态规划、贪心、回溯和分支界定

    算法系列之十六:使用穷举法解猜结果游戏--http://blog.csdn.net/orbit/article/details/7607685 ---------------例子: 麻将PC上发送操作 ...

  8. 五大常用算法一(回溯,随机化,动态规划)

    五大常用算法一(回溯,随机化,动态规划) 回溯算法 回溯法: 也称为试探法,它并不考虑问题规模的大小,而是从问题的最明显的最小规模开始逐步求解出可能的答案,并以此慢慢地扩大问题规模,迭代地逼近最终问题 ...

  9. html5走格子游戏,JS/HTML5游戏常用算法之碰撞检测 地图格子算法实例详解

    JS/HTML5游戏常用算法之碰撞检测 地图格子算法实例详解 发布时间:2020-09-26 20:42:24 来源:脚本之家 阅读:112 作者:krapnik 本文实例讲述了JS/HTML5游戏常 ...

最新文章

  1. F - Prime Path POJ - 3126
  2. OJ在线编程----常见输入输出练习场
  3. XenApp_XenDesktop_7.6实战篇之十四:XenDesktop虚拟桌面的交付
  4. 中科微研课题上犹授牌-农业大健康·李喜贵:谋定功能性农业
  5. c语言putchar_C语言实现变色的心!连机器都会变心,呵,男人!
  6. 百度AI学习:三、文字识别
  7. 安装 深度wine_深度系统更新(2020.11.25)
  8. VC MFC列表视图(CListCtrl)控件
  9. 比较linux的桌面环境的使用和异同?_深度UI + Ubuntu系统,堪称最强最美Linux发行版!你敢升级吗?...
  10. python input nameerror_用python写的,运行时出现NameError: name 'raw_input' is not defined 这是什么原因?求解...
  11. java可视化编程教程_JAVA可视化编程——SWING
  12. java项目-第35期在线购书商城系统【毕业设计】
  13. ioswifi信号测试软件app,‎App Store 上的“Speedtest - 网络测速测量和wifi分析仪”...
  14. 面试过程中应注意的问题与禁忌
  15. mac book air 装win7
  16. Windows HANDLE是什么
  17. 利用python修改Excel内容
  18. 安装pycrypto
  19. 看完涨薪2K+_____2019 JAVA 必刷面试题
  20. 狗年想有所成就?那你就要看这篇文章了

热门文章

  1. 计算机视觉的学习感悟
  2. mac识别不了u盘怎么办 mac系统格式化u盘win不识别怎么回事
  3. 浪潮服务器 np370d2 电源型号,2007盘点专稿:浪潮英信服务器NP370D
  4. H5页面验收流程及性能验收标准
  5. MyBatis项目报错java.io.IOException: Could not find resource mapping/UserMapper.xml
  6. linux多进程能同时执行吗,Linux开发简单多进程应用
  7. 思考:如何打造一个优秀的研发体系
  8. 疾病预测,机器学习和医疗保健
  9. Vue使用指南(一)
  10. 关于Thinkpad E495 指点杆(小红点)和触摸板同时失灵的解决方法