【优化数学模型】2. 动态规划DP方法的求解思路

2024-02-17 06:44

本文主要是介绍【优化数学模型】2. 动态规划DP方法的求解思路,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在这里插入图片描述

【优化数学模型】2. 动态规划DP方法的求解思路

  • 一、动态规划
    • 1. 概述
    • 2. 最优性原理
    • 3. 最优子结构特性
  • 二、示例:0-1背包问题
    • 1. 问题描述
    • 2. 使用自顶向下的方法
    • 3. 使用自下而上的方法


一、动态规划

1. 概述

多阶段决策问题,就是要在允许的决策范围内,选择一个最优决策使整个系统在预定标准下达到最佳效果。

动态规划 (dynamic programming,DP) 是用来解决多阶段决策过程最优化的一种数量方法。其特点在于,它可以把一个n维决策问题变换为几个一维最优化问题,从而一个一个地去解决。

与分而治之技术的不同之处在于,动态规划解决方案中的子问题是重叠的,因此解决一个子问题所需的一些相同步骤也需要用于其他子问题。

许多文章中提到动态规划会将其称为是一种经典的算法,需要指出的是,动态规划是1951年,美国数学家贝尔曼等人根据一类多阶段决策问题特性提出的解决这类问题的“最优化原理”,是其创建出的最优化问题的一种新思路,是求解一类问题的一种方法,是考察问题的一种途径,而不是一种算法。

在解决相应问题时,必须对具体问题进行具体分析,运用动态规划的原理和方法,建立相应的模型,然后再用动态规划方法去求解。

在这里插入图片描述

2. 最优性原理

应用动态规划设计使多阶段决策过程达到最优(成本最省、效益最高、路径最短等),依据动态规划的最优性原理:“作为整个过程的最优策略具有这样的性质,无论过去的状态和决策如何,对前面的决策所形成的状态而言,余下的诸决策必须构成最优策略”。也就是说,最优决策序列中的任何子序列都是最优的。

3. 最优子结构特性

最优性原理体现为问题的最优子结构特性。当一个问题的最优解中包含了子问题的最优解时,则称该问题具有最优子结构特性。最优子结构特性使得在从较小问题的解构造较大问题的解时,只需考虑子问题的最优解,从而大大减少了求解问题的计算量。

二、示例:0-1背包问题

1. 问题描述

0-1背包问题是应用动态规划设计求解的典型案例。

已知有 N 种物品和一个可容纳 W 重量的背包,物品 i 的重量为 w i w_i wi ,产生的效益为 p i p_i pi 。在装包时物品 i 可以装入,也可以不装,但不可拆开装。即物品 i可产生的效益为 x i p i x_ip_i xipi 。这里 x i x_i xi∈{0,1}, c, w i w_i wi , p i p_i pi∈ N+。

示例:
在这里插入图片描述

2. 使用自顶向下的方法

  • 一般使用一个数组或者一个哈希map充当这个备忘录。
  • 声明函数,并将项目、容量、临时字典和当前索引作为参数。
  • 如果小于 0、小于 0或大于列表长度,则返回 0.
  • 否则,如果总重量小于容量,则创建两个利润,用于检查相邻项目。这可以通过使用不同参数递归调用函数来实现。
  • 将这两个利润中的最大值分配给字典的当前元素。 将字典中的该值作为输出返回。
class Item:def __init__(self, profit, weight):self.profit = profitself.weight = weightdef zeroKnapsack(items, capacity, currentIndex, tempDict):dictKey = str(currentIndex) + str(capacity)if capacity <= 0 or currentIndex < 0 or currentIndex >= len(items):return 0elif dictKey in tempDict:return tempDict[currentIndex]elif items[currentIndex].weight <= capacity:profit1 = items[currentIndex].profit + zeroKnapsack(items, capacity-items[currentIndex].weight, currentIndex+1, tempDict)profit2 = zeroKnapsack(items, capacity, currentIndex+1, tempDict)tempDict[dictKey] = max(profit1, profit2)return tempDict[dictKey]else:return 0mango = Item(31, 3)
apple = Item(26, 1)
orange = Item(17, 2)
banana = Item(72, 5)items = [mango, apple, orange, banana]print(zeroKnapsack(items, 7, 0, {}))

3. 使用自下而上的方法

  • 声明函数,并将利润、物品权重和背包容量作为参数。
  • 如果容量小于 0,利润长度等于 0,或者利润长度和权重不一致,则返回 0。
  • 创建变量:迭代行数和容量,初始化的值为 0。
  • 创建两个变量用于存储利润。只要权重不超过容量,就存储矩阵中每一行和每一列的值。
  • 重复上述过程,并返回两个变量的最大值作为输出。
class Item:def __init__(self, profit, weight):self.profit = profitself.weight = weightdef zoKnapsackBU(profits, weights, capacity):if capacity <= 0 or len(profits) == 0 or len(weights) != len(profits):return 0numberOfRows = len(profits) + 1dp = [[None for i in range(capacity+2)] for j in range(numberOfRows)]for i in range(numberOfRows):dp[i][0] = 0for i in range(capacity+1):dp[numberOfRows-1][i] = 0for row in range(numberOfRows-2, -1, -1):for column in range(1, capacity+1):profit1 = 0profit2 = 0if weights[row] <= column:profit1 = profits[row] + dp[row + 1][column - weights[row]]profit2 = dp[row + 1][column]dp[row][column] = max(profit1, profit2)return dp[0][capacity]profits = [31, 26, 17, 72]
weights = [3, 1, 2, 5]print(zoKnapsackBU(profits, weights, 7))

最大利润是通过以下组合实现的:

Apple(W: 1, P: 26) + Banana(W: 5, P: 72) = W: 6, P: 98


这篇关于【优化数学模型】2. 动态规划DP方法的求解思路的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/716978

相关文章

Python安装Pandas库的两种方法

《Python安装Pandas库的两种方法》本文介绍了三种安装PythonPandas库的方法,通过cmd命令行安装并解决版本冲突,手动下载whl文件安装,更换国内镜像源加速下载,最后建议用pipli... 目录方法一:cmd命令行执行pip install pandas方法二:找到pandas下载库,然后

Linux系统中查询JDK安装目录的几种常用方法

《Linux系统中查询JDK安装目录的几种常用方法》:本文主要介绍Linux系统中查询JDK安装目录的几种常用方法,方法分别是通过update-alternatives、Java命令、环境变量及目... 目录方法 1:通过update-alternatives查询(推荐)方法 2:检查所有已安装的 JDK方

SQL Server安装时候没有中文选项的解决方法

《SQLServer安装时候没有中文选项的解决方法》用户安装SQLServer时界面全英文,无中文选项,通过修改安装设置中的国家或地区为中文中国,重启安装程序后界面恢复中文,解决了问题,对SQLSe... 你是不是在安装SQL Server时候发现安装界面和别人不同,并且无论如何都没有中文选项?这个问题也

Java Thread中join方法使用举例详解

《JavaThread中join方法使用举例详解》JavaThread中join()方法主要是让调用改方法的thread完成run方法里面的东西后,在执行join()方法后面的代码,这篇文章主要介绍... 目录前言1.join()方法的定义和作用2.join()方法的三个重载版本3.join()方法的工作原

go动态限制并发数量的实现示例

《go动态限制并发数量的实现示例》本文主要介绍了Go并发控制方法,通过带缓冲通道和第三方库实现并发数量限制,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面... 目录带有缓冲大小的通道使用第三方库其他控制并发的方法因为go从语言层面支持并发,所以面试百分百会问到

小白也能轻松上手! 路由器设置优化指南

《小白也能轻松上手!路由器设置优化指南》在日常生活中,我们常常会遇到WiFi网速慢的问题,这主要受到三个方面的影响,首要原因是WiFi产品的配置优化不合理,其次是硬件性能的不足,以及宽带线路本身的质... 在数字化时代,网络已成为生活必需品,追剧、游戏、办公、学习都离不开稳定高速的网络。但很多人面对新路由器

在MySQL中实现冷热数据分离的方法及使用场景底层原理解析

《在MySQL中实现冷热数据分离的方法及使用场景底层原理解析》MySQL冷热数据分离通过分表/分区策略、数据归档和索引优化,将频繁访问的热数据与冷数据分开存储,提升查询效率并降低存储成本,适用于高并发... 目录实现冷热数据分离1. 分表策略2. 使用分区表3. 数据归档与迁移在mysql中实现冷热数据分

Spring Boot从main方法到内嵌Tomcat的全过程(自动化流程)

《SpringBoot从main方法到内嵌Tomcat的全过程(自动化流程)》SpringBoot启动始于main方法,创建SpringApplication实例,初始化上下文,准备环境,刷新容器并... 目录1. 入口:main方法2. SpringApplication初始化2.1 构造阶段3. 运行阶

Olingo分析和实践之ODataImpl详细分析(重要方法详解)

《Olingo分析和实践之ODataImpl详细分析(重要方法详解)》ODataImpl.java是ApacheOlingoOData框架的核心工厂类,负责创建序列化器、反序列化器和处理器等组件,... 目录概述主要职责类结构与继承关系核心功能分析1. 序列化器管理2. 反序列化器管理3. 处理器管理重要方

Python错误AttributeError: 'NoneType' object has no attribute问题的彻底解决方法

《Python错误AttributeError:NoneTypeobjecthasnoattribute问题的彻底解决方法》在Python项目开发和调试过程中,经常会碰到这样一个异常信息... 目录问题背景与概述错误解读:AttributeError: 'NoneType' object has no at