带你读《图解算法小抄》二十一、动态规划(1)

简介: 带你读《图解算法小抄》二十一、动态规划(1)

二十一、动态规划

访问 www.coding-time.cn 阅读原文动画效果,体验更佳。


1.动态规划:备忘录和状态转移方程


动态规划(Dynamic Programming)是一种常用的算法设计和优化技术,广泛应用于解决各种优化问题。它的核心思想是将复杂问题分解为相互重叠的子问题,并通过记忆化搜索和状态转移来高效地求解问题。在动态规划中,备忘录和状态转移方程是两个重要的概念。

1备忘录

备忘录是动态规划中的一种优化技术,用于避免重复计算子问题。当我们解决一个问题时,往往会遇到许多重复的子问题。如果我们每次都重新计算这些子问题,那么计算量将会非常大。为了避免重复计算,我们可以使用备忘录来记录已经计算过的子问题的结果,以便在需要时直接查找并返回结果,而不必重新计算。

 

备忘录可以使用数组、哈希表或其他数据结构来存储子问题的结果。一般来说,我们可以使用一个一维或多维数组来构建备忘录。备忘录的初始化值通常是一个特殊的标识,表示该子问题尚未计算过。在计算子问题的过程中,我们可以先检查备忘录,如果发现该子问题已经计算过,直接返回备忘录中的结果;否则,我们进行计算,并将结果存入备忘录中。

 

下面是一个通用的备忘录模板:

 

function memoizedDP(...) {
  // 初始化备忘录
  const memo = new Array(...);
  function dp(...) {
    if (已经计算过该子问题) {
      return 从备忘录中返回结果;
    }    // 计算子问题
    const result = ...;    // 将结果存入备忘录
memo[...] = result;
    return result;
  }
  return dp(...);
}

2状态转移方程

状态转移方程是动态规划中的另一个重要概念,用于描述子问题之间的关系。它是动态规划问题的核心思想之一。通过定义状态转移方程,我们可以将原始问题划分为若干个子问题,并找到它们之间的递推关系。

 

状态转移方程通常用数学公式或逻辑表达式表示。它描述了问题的当前状态和下一个状态之间的转移方式。通过递推计算,我们可以从初始状态逐步推导出最终的解。

 

在使用状态转移方程求解问题时,一般需要注意以下几点:

 

  • 定义状态:确定问题的状态,即表示问题的变量或参数。状态可以是一维、二维甚至多维的,具体根据问题的特点来确定。
  • 初始状态:确定问题的初始状态,即起始条件。
  • 状态转移方程:根据问题的递推关系,描述当前状态和下一个状态之间的转移方式。这通常是问题的核心部分,需要仔细分析问题的特点和规律。
  • 边界条件:定义问题的边界条件,即最小规模的子问题的解。边界条件通常是已知的,可以直接计算得出。

3动态规划算法框架

动态规划算法可以用以下框架来解决问题:

function dynamicProgramming(...) {
  // 初始化备忘录
  const memo = new Array(...);  function dp(...) {
    // 边界条件
    if (达到边界条件) {
      return 边界条件的结果;
    }    // 检查备忘录
    if (已经计算过该子问题) {
      return 从备忘录中返回结果;
    }    // 根据状态转移方程计算子问题
    const result = 根据状态转移方程计算;    // 将结果存入备忘录
memo[...] = result;
    return result;
  }
  return dp(...);
}

通过以上算法框架,我们可以将原始问题划分为多个子问题,并通过备忘录和状态转移方程高效地求解问题。其中,备忘录用于避免重复计算子问题,而状态转移方程描述了子问题之间的关系,帮助我们逐步求解最终的解。


带你读《图解算法小抄》二十一、动态规划(2)https://developer.aliyun.com/article/1347964

相关文章
|
5月前
|
机器学习/深度学习 存储 算法
动态规划算法深度解析:0-1背包问题
0-1背包问题是经典的组合优化问题,目标是在给定物品重量和价值及背包容量限制下,选取物品使得总价值最大化且每个物品仅能被选一次。该问题通常采用动态规划方法解决,通过构建二维状态表dp[i][j]记录前i个物品在容量j时的最大价值,利用状态转移方程避免重复计算子问题,从而高效求解最优解。
672 1
|
12月前
|
存储 算法 Java
算法系列之动态规划
动态规划(Dynamic Programming,简称DP)是一种用于解决复杂问题的算法设计技术。它通过将问题分解为更小的子问题,并存储这些子问题的解来避免重复计算,从而提高算法的效率。
463 4
算法系列之动态规划
|
算法 Java C++
【潜意识Java】蓝桥杯算法有关的动态规划求解背包问题
本文介绍了经典的0/1背包问题及其动态规划解法。
462 5
|
12月前
|
算法 安全 调度
【动态规划篇】穿越算法迷雾:约瑟夫环问题的奇幻密码
【动态规划篇】穿越算法迷雾:约瑟夫环问题的奇幻密码
|
12月前
|
机器学习/深度学习 算法 测试技术
【动态规划篇】01 背包的逆袭:如何用算法装满你的 “财富背包”
【动态规划篇】01 背包的逆袭:如何用算法装满你的 “财富背包”
|
算法
动态规划算法学习三:0-1背包问题
这篇文章是关于0-1背包问题的动态规划算法详解,包括问题描述、解决步骤、最优子结构性质、状态表示和递推方程、算法设计与分析、计算最优值、算法实现以及对算法缺点的思考。
749 2
动态规划算法学习三:0-1背包问题
|
算法 Python
在Python编程中,分治法、贪心算法和动态规划是三种重要的算法。分治法通过将大问题分解为小问题,递归解决后合并结果
在Python编程中,分治法、贪心算法和动态规划是三种重要的算法。分治法通过将大问题分解为小问题,递归解决后合并结果;贪心算法在每一步选择局部最优解,追求全局最优;动态规划通过保存子问题的解,避免重复计算,确保全局最优。这三种算法各具特色,适用于不同类型的问题,合理选择能显著提升编程效率。
305 2
|
算法
动态规划算法学习四:最大上升子序列问题(LIS:Longest Increasing Subsequence)
这篇文章介绍了动态规划算法中解决最大上升子序列问题(LIS)的方法,包括问题的描述、动态规划的步骤、状态表示、递推方程、计算最优值以及优化方法,如非动态规划的二分法。
339 0
动态规划算法学习四:最大上升子序列问题(LIS:Longest Increasing Subsequence)
|
算法
动态规划算法学习二:最长公共子序列
这篇文章介绍了如何使用动态规划算法解决最长公共子序列(LCS)问题,包括问题描述、最优子结构性质、状态表示、状态递归方程、计算最优值的方法,以及具体的代码实现。
725 0
动态规划算法学习二:最长公共子序列
|
存储 算法
动态规划算法学习一:DP的重要知识点、矩阵连乘算法
这篇文章是关于动态规划算法中矩阵连乘问题的详解,包括问题描述、最优子结构、重叠子问题、递归方法、备忘录方法和动态规划算法设计的步骤。
1392 0