带你读《图解算法小抄》二十一、动态规划(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

相关文章
|
3月前
|
存储 算法
深入了解动态规划算法
深入了解动态规划算法
82 1
|
3月前
|
算法 测试技术 C++
【动态规划算法】蓝桥杯填充问题(C/C++)
【动态规划算法】蓝桥杯填充问题(C/C++)
|
2月前
|
算法 Python
在Python编程中,分治法、贪心算法和动态规划是三种重要的算法。分治法通过将大问题分解为小问题,递归解决后合并结果
在Python编程中,分治法、贪心算法和动态规划是三种重要的算法。分治法通过将大问题分解为小问题,递归解决后合并结果;贪心算法在每一步选择局部最优解,追求全局最优;动态规划通过保存子问题的解,避免重复计算,确保全局最优。这三种算法各具特色,适用于不同类型的问题,合理选择能显著提升编程效率。
61 2
|
3月前
|
算法
动态规划算法学习三:0-1背包问题
这篇文章是关于0-1背包问题的动态规划算法详解,包括问题描述、解决步骤、最优子结构性质、状态表示和递推方程、算法设计与分析、计算最优值、算法实现以及对算法缺点的思考。
113 2
动态规划算法学习三:0-1背包问题
|
3月前
|
算法
动态规划算法学习四:最大上升子序列问题(LIS:Longest Increasing Subsequence)
这篇文章介绍了动态规划算法中解决最大上升子序列问题(LIS)的方法,包括问题的描述、动态规划的步骤、状态表示、递推方程、计算最优值以及优化方法,如非动态规划的二分法。
81 0
动态规划算法学习四:最大上升子序列问题(LIS:Longest Increasing Subsequence)
|
3月前
|
算法
动态规划算法学习二:最长公共子序列
这篇文章介绍了如何使用动态规划算法解决最长公共子序列(LCS)问题,包括问题描述、最优子结构性质、状态表示、状态递归方程、计算最优值的方法,以及具体的代码实现。
192 0
动态规划算法学习二:最长公共子序列
|
3月前
|
存储 人工智能 算法
【算法——动态规划】蓝桥ALGO-1007 印章(C/C++)
【算法——动态规划】蓝桥ALGO-1007 印章(C/C++)
【算法——动态规划】蓝桥ALGO-1007 印章(C/C++)
|
3月前
|
存储 算法
动态规划算法学习一:DP的重要知识点、矩阵连乘算法
这篇文章是关于动态规划算法中矩阵连乘问题的详解,包括问题描述、最优子结构、重叠子问题、递归方法、备忘录方法和动态规划算法设计的步骤。
187 0
|
3月前
|
算法 C++
【算法解题思想】动态规划+深度优先搜索(C/C++)
【算法解题思想】动态规划+深度优先搜索(C/C++)
|
5月前
|
机器学习/深度学习 算法 Java
算法设计(动态规划应用实验报告)实现基于贪婪技术思想的Prim算法、Dijkstra算法
这篇文章介绍了基于贪婪技术思想的Prim算法和Dijkstra算法,包括它们的伪代码描述、Java源代码实现、时间效率分析,并展示了算法的测试用例结果,使读者对贪婪技术及其应用有了更深入的理解。
算法设计(动态规划应用实验报告)实现基于贪婪技术思想的Prim算法、Dijkstra算法

热门文章

最新文章