动态规划——切割钢条问题

2024-04-24 05:04

本文主要是介绍动态规划——切割钢条问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、动态规划

       动态规划算法通常用于解决最优化问题寻求最优解)。其思想与分治法类似,将待求解的问题分成若干个子问题,先求出子问题,再根据子问题的解求出原来问题中的解,与分支法不同的是,在动态规划中,这些子问题的解是不相互独立的

        采用动态规划求解的问题通常有以下性质:

        1.最优化原理:问题的最优解中包含的子问题的解也是最优的

        2.无后效性:即某阶段状态一旦确定,就不受该状态以后决策的影响,只与当前状态有关。

        3.有重叠子问题:子问题之间是不相互独立的,一个子问题在下一阶段的决策中可能被多次用到。(通常减少不必要的重复操作)

二、钢条切割问题        

        给定一段长度为 n 的钢条和一个价格表 pi ,求钢条切割方案使得销售收益 rn 最大。

长度I

1

2

3

4

5

6

7

8

9

10

价格pi

1

5

8

9

10

17

17

20

24

30

        案例:长度为 4 的钢条,如何切割使得销售额最高?

        考虑两种情况:1.切割成4个长度为1的钢条,总收益是4;

                                  2.切割成2个长度为2的钢条,总收益是10。

                                  3.切割成1个长度为1和一个长度为3的钢条,总收益是9

                                 4.切割成1个长度为4的钢条,总收益为9

        法一(易理解)

        对于求收益r[n]最大的切割方案(最优解)

        1.不切割,收益为 pn

        2.先将该钢条分为切为两根,则当该两根钢条的收益之和最大时(取最优解时),对应长度为n的钢条收益也最大,最优解的和就是当前情况的最优解,可以得出:

                r_{n}=max(p_{n},r_{n-1} +r_{1},r_{n-2} +r_{2},\cdots ,r_{1} +r_{n-1}) 

        法二 (简单)

        对于该求解方法可以改为一种相似但更简单的递归求解方法:

        将钢条从左边切割下长度为 i 的一段,只对右边剩下的长度为 n-i 的一段继续进行切割(递归求解),对左边的一段不再进行切割。

关于此想法的理解:

将长度为n的钢条分解为左边开始一段,以及剩余部分继续分解的结果(通过递归) 

或者理解为:对于一根长度为n钢条,总存在某种切割,会使得切出长度为 i 的钢条  (1\leq i\leq n)

           此时公式为:

                                r_{n}=\underset{1\leq i\leq n}{max}(p_{i}+r_{n-i})

           递归函数的伪代码为:

int get_best(int n)
{if(n<=0) return 0;int maxn=-1;for(int i=1;i<=n;i++)maxn=max(p[i]+get_best(n-i),maxn);  // 通过递归求出最大//r[n]=maxn;return maxn;
}

        根据代码不难发现,在递归函数get_best中,会存在同一个变量反复递归的情况,从而引起时间的浪费,此时时间复杂度达到 O(2^n)

        需要通过剪枝的方法避免重复的操作(自顶向下法)

int get_best(int n)
{if(n<=0) return 0;if(r[n]>0) return r[n]; // 若已经访问过,即找到r[n]的最优解时,直接返回int maxn=-1;for(int i=1;i<=n;i++)maxn=max(p[i]+get_best(n-i),maxn);r[n]=maxn;return maxn;
}

  此外,还可以通过自底向上的方法求出最优解,此时为递推操作,不需要递归

int get_best2(int n)
{for(int i=1;i<=n;i++){r[i]=p[i]; // 直接将 长度为 i的切割下 for(int j=1;j<i;j++)r[i]=max(r[i],p[j]+r[i-j]); //免去递归操作 }} 

         重构解

        将最优解的切割方案求出

int get_best2(int n)
{for(int i=1;i<=n;i++){r[i]=p[i]; // 直接将 长度为 i的切割下 s[i]=i;for(int j=1;j<i;j++)if(r[i]<p[j]+r[i-j])  //免去递归操作 {r[i]=p[j]+r[i-j];s[i]=j;  // 表示当长度为i时,将切割长度为j的钢条 ,剩余 i-j 已经求出最优解和切割方案了 }}int x=n;while(x>0){printf("%d ",s[x]);x-=s[x];}
}

此过程只需在求解规模为 i 的子问题时将,第一段钢条的最优切割长度j保存在 s [ i ] 中

这篇关于动态规划——切割钢条问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

MySQL 设置AUTO_INCREMENT 无效的问题解决

《MySQL设置AUTO_INCREMENT无效的问题解决》本文主要介绍了MySQL设置AUTO_INCREMENT无效的问题解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参... 目录快速设置mysql的auto_increment参数一、修改 AUTO_INCREMENT 的值。

关于跨域无效的问题及解决(java后端方案)

《关于跨域无效的问题及解决(java后端方案)》:本文主要介绍关于跨域无效的问题及解决(java后端方案),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录通用后端跨域方法1、@CrossOrigin 注解2、springboot2.0 实现WebMvcConfig

Go语言中泄漏缓冲区的问题解决

《Go语言中泄漏缓冲区的问题解决》缓冲区是一种常见的数据结构,常被用于在不同的并发单元之间传递数据,然而,若缓冲区使用不当,就可能引发泄漏缓冲区问题,本文就来介绍一下问题的解决,感兴趣的可以了解一下... 目录引言泄漏缓冲区的基本概念代码示例:泄漏缓冲区的产生项目场景:Web 服务器中的请求缓冲场景描述代码

Java死锁问题解决方案及示例详解

《Java死锁问题解决方案及示例详解》死锁是指两个或多个线程因争夺资源而相互等待,导致所有线程都无法继续执行的一种状态,本文给大家详细介绍了Java死锁问题解决方案详解及实践样例,需要的朋友可以参考下... 目录1、简述死锁的四个必要条件:2、死锁示例代码3、如何检测死锁?3.1 使用 jstack3.2

解决JSONField、JsonProperty不生效的问题

《解决JSONField、JsonProperty不生效的问题》:本文主要介绍解决JSONField、JsonProperty不生效的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录jsONField、JsonProperty不生效javascript问题排查总结JSONField

github打不开的问题分析及解决

《github打不开的问题分析及解决》:本文主要介绍github打不开的问题分析及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、找到github.com域名解析的ip地址二、找到github.global.ssl.fastly.net网址解析的ip地址三

MySQL版本问题导致项目无法启动问题的解决方案

《MySQL版本问题导致项目无法启动问题的解决方案》本文记录了一次因MySQL版本不一致导致项目启动失败的经历,详细解析了连接错误的原因,并提供了两种解决方案:调整连接字符串禁用SSL或统一MySQL... 目录本地项目启动报错报错原因:解决方案第一个:第二种:容器启动mysql的坑两种修改时区的方法:本地

springboot加载不到nacos配置中心的配置问题处理

《springboot加载不到nacos配置中心的配置问题处理》:本文主要介绍springboot加载不到nacos配置中心的配置问题处理,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录springboot加载不到nacos配置中心的配置两种可能Spring Boot 版本Nacos

Java调用C#动态库的三种方法详解

《Java调用C#动态库的三种方法详解》在这个多语言编程的时代,Java和C#就像两位才华横溢的舞者,各自在不同的舞台上展现着独特的魅力,然而,当它们携手合作时,又会碰撞出怎样绚丽的火花呢?今天,我们... 目录方法1:C++/CLI搭建桥梁——Java ↔ C# 的“翻译官”步骤1:创建C#类库(.NET

Java中JSON格式反序列化为Map且保证存取顺序一致的问题

《Java中JSON格式反序列化为Map且保证存取顺序一致的问题》:本文主要介绍Java中JSON格式反序列化为Map且保证存取顺序一致的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未... 目录背景问题解决方法总结背景做项目涉及两个微服务之间传数据时,需要提供方将Map类型的数据序列化为co