算法专题1——动态规划 Dynamic Programming,DP

简介: 算法专题1——动态规划 Dynamic Programming,DP

DP学一次忘一次,干脆写个总结以后好复习

原文发布时间:2022-09-29 16:47:04

本文主要为,代码随想录的学习笔记

质量分太低,进行扩展补充

原文

一、What

如果某⼀问题有很多重叠⼦问题,使⽤动态规划

是最有效的。

由dp[j-weight[i]]推导出来的,然后取max(dp[j], dp[j - weight[i]] + value[i])。

所以动态规划中每⼀个状态⼀定是由上一个状态推导出来的

区分于贪心

贪心没有状态推导,⽽是从局部直接选最优的

二、解题步骤

  1. 确定dp数组(dp table)以及下标的含义
  2. 确定递推公式
  3. dp数组如何初始化
  4. 确定遍历顺序
  5. 举例推导dp数组

三、如何debug

打印dp数组

写代码前将状态转移在dp数组上具体模拟一遍,确认是想要的结果

这道题目我举例推导状态转移公式了么?

我打印dp数组的日志了么?

打印出来了dp数组和我想的⼀样么?

完善补充

探秘DP算法的工作原理

动态规划(Dynamic Programming)是一种强大的算法技术,用于解决各种计算问题,包括优化、最短路径、序列匹配、资源分配等等。本文将深入探讨动态规划算法的工作原理,以及它在解决各种问题中的应用。

什么是动态规划?

动态规划是一种通过将问题分解为子问题并将它们的解决方案存储在表格中以避免重复计算的算法技术。它通常适用于满足以下两个条件的问题:

  1. 最优子结构:问题的最优解可以通过子问题的最优解来构建。
  2. 重叠子问题:问题可以分解为多个重叠的子问题,这些子问题可以在不同上下文中多次出现。

动态规划通过解决子问题来构建整个问题的解,从而避免了不必要的重复计算,提高了算法的效率。

动态规划的基本原理

动态规划算法通常遵循以下基本原理:

  1. 问题拆分:将原始问题分解为一系列子问题,这些子问题通常是相互重叠的。
  2. 状态定义:明确定义每个子问题的状态。状态是问题的一个关键属性,用于描述问题的某一方面。
  3. 状态转移方程:确定如何从一个状态转移到另一个状态。这通常是问题的核心部分,定义了问题的解决方案。
  4. 初始化:设置初始状态的值,通常是最简单的情况的解。
  5. 自底向上求解:使用状态转移方程自底向上地求解子问题,直到解决了整个问题。

动态规划的经典问题

动态规划算法广泛应用于各种问题,以下是一些经典的示例:

  1. 斐波那契数列:通过动态规划可以高效地计算斐波那契数列的第n项,而不需要递归。
  2. 最长公共子序列:用于比较两个字符串,找到它们的最长公共子序列。
  3. 最短路径问题:例如,Dijkstra算法和Floyd-Warshall算法可以使用动态规划来解决最短路径问题。
  4. 背包问题:用于在给定容量的情况下,选择一组物品以最大化其价值,而不超过容量。
  5. 编辑距离:用于计算将一个字符串转换为另一个字符串所需的最少编辑操作数。

动态规划的优缺点

动态规划算法的优点包括:

  • 高效性:通过避免重复计算,可以显著提高问题的解决效率。
  • 通用性:适用于解决各种问题,从计算问题到优化问题。

然而,动态规划算法也有一些缺点:

  • 需要额外的内存:为了存储子问题的解,需要额外的内存空间。
  • 需要定义状态和状态转移方程:有时难以确定问题的状态和状态转移方程。

代码案例

注意:动态规划的实现方式会因问题而异,但通常都包括问题的拆分、状态定义、状态转移方程、初始化等步骤。

案例1:斐波那契数列

动态规划问题的具体实现代码会根据不同的问题而异。以下是一个经典的动态规划问题示例——斐波那契数列的Python代码:

def fibonacci(n):
    fib = [0] * (n + 1)
    fib[0] = 0
    fib[1] = 1
    for i in range(2, n + 1):
        fib[i] = fib[i - 1] + fib[i - 2]
    return fib[n]
n = 10
result = fibonacci(n)
print(f"The {n}-th Fibonacci number is {result}")

这个代码用动态规划来计算斐波那契数列的第n个数,避免了递归的重复计算。

案例2:0/1背包问题

理解动态规划的概念和应用需要逐渐深入,因此我们将探讨一个稍微复杂一点的问题:0/1背包问题(0/1 Knapsack Problem)

这是一个经典的动态规划问题,涉及如何在给定容量的情况下选择一组物品,以最大化它们的总价值,同时保持总重量不超过容量

以下是一个Python示例代码,解决0/1背包问题:

def knapsack(values, weights, capacity):
    n = len(values)
    dp = [[0] * (capacity + 1) for _ in range(n + 1)]
    for i in range(n + 1):
        for w in range(capacity + 1):
            if i == 0 or w == 0:
                dp[i][w] = 0
            elif weights[i - 1] <= w:
                dp[i][w] = max(dp[i - 1][w], dp[i - 1][w - weights[i - 1]] + values[i - 1])
            else:
                dp[i][w] = dp[i - 1][w]
    return dp[n][capacity]
# 示例
values = [60, 100, 120]
weights = [10, 20, 30]
capacity = 50
result = knapsack(values, weights, capacity)
print(f"最大总价值为 {result}")

上述代码使用动态规划来解决0/1背包问题。knapsack函数接受物品的价值列表values、物品的重量列表weights以及背包的容量capacity作为输入,返回最大总价值。它使用二维数组dp来保存子问题的解决方案,然后通过填充表格中的值来逐步计算出最优解。

这个示例展示了如何使用动态规划来解决更复杂的问题,动态规划的核心思想是将问题分解为子问题,通过解决子问题来构建最终的解决方案。

希望这个示例能帮助更好地理解动态规划的实际应用。

案例3:最长递增子序列问题

当涉及更复杂的动态规划问题时,问题的实现会更加复杂。以下是一个稍复杂的示例:最长递增子序列(Longest Increasing Subsequence, LIS)

这个问题的目标是:找到给定序列中的最长递增子序列

下面是一个用Python解决LIS问题的示例代码:

def longest_increasing_subsequence(arr):
    n = len(arr)
    lis = [1] * n
    for i in range(1, n):
        for j in range(0, i):
            if arr[i] > arr[j] and lis[i] < lis[j] + 1:
                lis[i] = lis[j] + 1
    max_length = max(lis)
    return max_length
# 示例
sequence = [10, 22, 9, 33, 21, 50, 41, 60, 80]
result = longest_increasing_subsequence(sequence)
print(f"最长递增子序列的长度为 {result}")

longest_increasing_subsequence函数接受一个整数序列作为输入,并返回最长递增子序列的长度。它使用一个列表lis来跟踪每个元素的LIS长度,然后通过填充表格中的值来计算最长递增子序列的长度。

这是一个相对较复杂的动态规划问题,需要利用子问题的解构建最终的解。动态规划的复杂性通常随着问题的难度而增加。

结语

动态规划是一种强大的算法技术,可以应用于各种计算问题。

了解动态规划的基本原理和应用领域可以帮助您更好地解决复杂的问题,提高算法的效率。

希望这篇文章能帮助您更好地理解和应用动态规划算法。

目录
相关文章
|
1月前
|
算法 Python
在Python编程中,分治法、贪心算法和动态规划是三种重要的算法。分治法通过将大问题分解为小问题,递归解决后合并结果
在Python编程中,分治法、贪心算法和动态规划是三种重要的算法。分治法通过将大问题分解为小问题,递归解决后合并结果;贪心算法在每一步选择局部最优解,追求全局最优;动态规划通过保存子问题的解,避免重复计算,确保全局最优。这三种算法各具特色,适用于不同类型的问题,合理选择能显著提升编程效率。
48 2
|
2月前
|
算法
动态规划算法学习三:0-1背包问题
这篇文章是关于0-1背包问题的动态规划算法详解,包括问题描述、解决步骤、最优子结构性质、状态表示和递推方程、算法设计与分析、计算最优值、算法实现以及对算法缺点的思考。
97 2
动态规划算法学习三:0-1背包问题
|
2月前
|
算法
动态规划算法学习四:最大上升子序列问题(LIS:Longest Increasing Subsequence)
这篇文章介绍了动态规划算法中解决最大上升子序列问题(LIS)的方法,包括问题的描述、动态规划的步骤、状态表示、递推方程、计算最优值以及优化方法,如非动态规划的二分法。
79 0
动态规划算法学习四:最大上升子序列问题(LIS:Longest Increasing Subsequence)
|
2月前
|
算法
动态规划算法学习二:最长公共子序列
这篇文章介绍了如何使用动态规划算法解决最长公共子序列(LCS)问题,包括问题描述、最优子结构性质、状态表示、状态递归方程、计算最优值的方法,以及具体的代码实现。
166 0
动态规划算法学习二:最长公共子序列
|
2月前
|
存储 算法
动态规划算法学习一:DP的重要知识点、矩阵连乘算法
这篇文章是关于动态规划算法中矩阵连乘问题的详解,包括问题描述、最优子结构、重叠子问题、递归方法、备忘录方法和动态规划算法设计的步骤。
150 0
|
15天前
|
算法
基于WOA算法的SVDD参数寻优matlab仿真
该程序利用鲸鱼优化算法(WOA)对支持向量数据描述(SVDD)模型的参数进行优化,以提高数据分类的准确性。通过MATLAB2022A实现,展示了不同信噪比(SNR)下模型的分类误差。WOA通过模拟鲸鱼捕食行为,动态调整SVDD参数,如惩罚因子C和核函数参数γ,以寻找最优参数组合,增强模型的鲁棒性和泛化能力。
|
21天前
|
机器学习/深度学习 算法 Serverless
基于WOA-SVM的乳腺癌数据分类识别算法matlab仿真,对比BP神经网络和SVM
本项目利用鲸鱼优化算法(WOA)优化支持向量机(SVM)参数,针对乳腺癌早期诊断问题,通过MATLAB 2022a实现。核心代码包括参数初始化、目标函数计算、位置更新等步骤,并附有详细中文注释及操作视频。实验结果显示,WOA-SVM在提高分类精度和泛化能力方面表现出色,为乳腺癌的早期诊断提供了有效的技术支持。
|
1天前
|
供应链 算法 调度
排队算法的matlab仿真,带GUI界面
该程序使用MATLAB 2022A版本实现排队算法的仿真,并带有GUI界面。程序支持单队列单服务台、单队列多服务台和多队列多服务台三种排队方式。核心函数`func_mms2`通过模拟到达时间和服务时间,计算阻塞率和利用率。排队论研究系统中顾客和服务台的交互行为,广泛应用于通信网络、生产调度和服务行业等领域,旨在优化系统性能,减少等待时间,提高资源利用率。
|
8天前
|
存储 算法
基于HMM隐马尔可夫模型的金融数据预测算法matlab仿真
本项目基于HMM模型实现金融数据预测,包括模型训练与预测两部分。在MATLAB2022A上运行,通过计算状态转移和观测概率预测未来值,并绘制了预测值、真实值及预测误差的对比图。HMM模型适用于金融市场的时间序列分析,能够有效捕捉隐藏状态及其转换规律,为金融预测提供有力工具。
|
17天前
|
算法
基于GA遗传算法的PID控制器参数优化matlab建模与仿真
本项目基于遗传算法(GA)优化PID控制器参数,通过空间状态方程构建控制对象,自定义GA的选择、交叉、变异过程,以提高PID控制性能。与使用通用GA工具箱相比,此方法更灵活、针对性强。MATLAB2022A环境下测试,展示了GA优化前后PID控制效果的显著差异。核心代码实现了遗传算法的迭代优化过程,最终通过适应度函数评估并选择了最优PID参数,显著提升了系统响应速度和稳定性。