动态规划之背包问题

上传人:桔**** 文档编号:561282509 上传时间:2022-08-10 格式:DOCX 页数:5 大小:14.74KB
返回 下载 相关 举报
动态规划之背包问题_第1页
第1页 / 共5页
动态规划之背包问题_第2页
第2页 / 共5页
动态规划之背包问题_第3页
第3页 / 共5页
动态规划之背包问题_第4页
第4页 / 共5页
动态规划之背包问题_第5页
第5页 / 共5页
亲,该文档总共5页,全部预览完了,如果喜欢就下载吧!
资源描述

《动态规划之背包问题》由会员分享,可在线阅读,更多相关《动态规划之背包问题(5页珍藏版)》请在金锄头文库上搜索。

1、动态规划动态规划的逆向思维法 动态规划是一种思维方法,没有统一的、具体的模式。动态规划可以从多方面去考察,不同的方面对动态规划有不同的表述。我们不打算强加一种统一 的表述,而是从多个角度对动态规划的思维方法进行讨论,希望大家在思维具体问题时,也能够从多个角度展开,这样收获会更大。逆向思维法是指从问题目标状态出发倒推回初始状态或边界状态的思维方法。如果原问题可以分解成几个本质相同、规模较小的问题,很自然就会 联想到从逆向思维的角度寻求问题的解决。你也许会想,这种将大问题分解成小问题的思维不就是分治法吗?动态规划是不是分而治之呢?其实,虽然我们在运用动态规划的逆向思维法和分治 法分析问题时,都使用

2、了这种将问题实例归纳为更小的、相似的子问题,并通过求解子问题产生一个全局最优值的思路,但动态规划不是分治法:关键 在于分解出来的各个子问题的性质不同。分治法要求各个子问题是独立的(即不包含公共的子问题),因此一旦递归地求出各个子问题的解后,便可自下而上地将子问题的解合并成原问题的解。 如果各子问题是不独立的,那么分治法就要做许多不必要的工作,重复地解公共的子问题。动态规划与分治法的不同之处在于动态规划允许这些子问题不独立(即各子问题可包含公共的子问题),它对每个子问题只解一次,并将结果保存起来, 避免每次碰到时都要重复计算。这就是动态规划高效的一个原因。动态规划的逆向思维法的要点可归纳为以下三

3、个步骤:(1) 分析最优值的结构,刻画其结构特征;(2) 递归地定义最优值;(3) 按自底向上或自顶向下记忆化的方式计算最优值。【例题 5】背包问题描述:有一个负重能力为m的背包和n种物品,第i种物品的价值为vi,重量为wi。在不超过背包负重能力的前提下选择若干个物品装入背包,使这 些的物品的价值之和最大。每种物品可以不选,也可以选择多个。假设每种物品都有足够的数量。分析: 从算法的角度看,解决背包问题一种最简单的方法是枚举所有可能的物品的组合方案并计算这个组合方案的价值之和,从中找出价值之和最大的方 案。显然,这种靠穷举所有可能方案的方法不是一种有效的算法。但是这个问题可以使用动态规划加以解

4、决。下面我们用动态规划的逆向思维法来分析这个问题。(1) 背包问题最优值的结构 动态规划的逆向思维法的第一步是刻画一个最优值的结构,如果我们能分析出一个问题的最优值包含其子问题的最优值,问题的这种性质称为最优子结构。一个问题的最优子结构性质是该问题可以使用动态规划的显著特征。对一个负重能力为m的背包,如果我们选择装入一个第i种物品,那么原背包问题就转化为负重能力为m-wi的子背包问题。原背包问题的最 优值包含这个子背包问题的最优值。若我们用背包的负重能力来划分状态,令状态变量sk表示负重能力为k的背包,那么sm 的值只取决于sk(km) 的值。因此背包问题具有最优子结构。(2) 递归地定义最优

5、值动态规划的逆向思维法的第二步是根据各个子问题的最优值来递归地定义原问题的最优值。对背包问题而言,有状态转移方程:/ maxsk-wi+vi(其中 1i0)sk= 若 k0 且存在 1i0, 0 否则。有了计算各个子冋题旳最优值旳递归式,我们就可以直接编与对应旳程丿予。卜述旳函数knapsack是输人背包旳负重能力k 返回对应旳子背包冋题的最优值sk:function knapsack(k:integer):integer;beginknapsack:=0;for i:=1 to n doif k-wi=0 thenbegint:=knapsack(k-wi)+vi;if knapsack t

6、 then knapsack:=t;end;end;上述递归算法在求解过程中反复出现了一个子问题,且对每次重复出现的子问题都要重新解一次,这需要多花费不少时间。卜面先考虑一个具体的背包问题。例如,当m=3, n=2, v1=1, w1=1, v2=2, w2=2,图1示出了由调用knapsack(3)所产生的递归树,每一个结点上标有参数k的值,请注意某些数出现了多次。3/21 1 0 0/0图1例如,knapsack(1)被引用了两次:在计算knapsack(3)和knapsack(2)中分别被引用;而knapsack(O)更是被引用了三次。如果knapsack(1)和knapsack(O)每

7、次都要被重新计算,则增加的运行时间相当可观。卜面,我们来看看动态规划是如何解决这个问题的。(3) 按自顶向卜记忆化或自底向上的方式求最优解一般地,如果一个解最优化问题的递归算法经常反复地解重复的子问题,而不是总在产生新的子问题时,我们说该最优化问题包含重迭子问题。这 类问题不宜用分治法求解,因为分治法递归的每一步要求产生相异的子问题。在这种情况卜,采用动态规划是很合适的,因为该方法对每一个子问题只 解一次,然后把解存放在一个表中,以便在解同样的子问题时查阅,充分利用了重迭子问题。一般来说,解决重迭子问题的方式有两种。自顶向卜的记忆化方式该方式的程予流程基本按照原问题的递归定义,不同的是,它专门

8、设置了一张表,以记忆在求解过程中得出的所有子问题的解。一个记忆的递归算法为每个子问题的解在表中记录一个表项。初始时每个表项都包含一个特殊值,以示该表项的解有待填入。例如背包问题中:si = -1; (0im)当在递归算法的执行中第一次遇到一个子问题时(即si=-1),计算其解并填入表中,以后每遇到该子问题,只要查看表中先前填入的值即可。 下面,我们按照自上而下的记忆化方式,写出求背包问题的递归算法。函数memorized_knapsack输入背包的负重能力k,返回背包问题的解sk。s表应设为全局变量,使得函数执行后,传出所有子问题的解si(0i=0 thenbegint:=memorized_

9、knapsack(k-Wi)+Vi;if sk t then sk:=t;end;end;memorized_knapsack:=sk;end;我们在主程序通过调用memorized_knapsack(m)即可获得背包问题的解。显然这种自顶向下记忆化的算法效率比重复解重迭子问题的knapsack算法要强一些。此外,在程序中加入判别哪些子问题需要求解的语句,只解 那些肯定要解的子问题,从而节省了时间。自底向上的方式假设我们要解决在分析函数knapsack当中提出的那个具体的背包问题。观察一下在调用mem or ized_knapsack(4)的过程中,sk 被计算出来的 顺序。我们发现,首先是s

10、0被计算出来,然后是s1,s2,最后s3被计算出来,这与调用的次序正好相反。动态规划的执行方式是自底向上,所有的子问题计算一次,充分利用重迭子问题。因此,我们很自然就想到与这种自底向上的执行方式相应的采用 自底向上的方式递推所有子问题的最优值。我们知道,计算负重能力为k的背包问题的解仅依赖于计算负重能力小于k的背包问题的解,因此填s表的方式与解决负重能力递增的背包问题相 对应:最初设定负重能力为0的背包问题的解s0 = 0o然后依次考虑负重能力为1, 2,,m的背包问题的解。每填入一个sk(0km)时,充分利用所有重迭子问题的解:sk-wi(0i0)下面是按照自底向上方式计算背包问题的算法流程

11、。过程knapsack_order输入背包的负重能力k,s表设为全局变量。过程执行后所有子问题的 解通过s表传给主程序。procedure knapsack_order(k:integer);beginfor i:=1 to k do si:=0;for j:=1 to k dofor i:=1 to n doif j-wi=0 thenbegint:=sj-wi+vi;if sj =0 thenbegint:=sj-wi+vi;if sj t then sj:=t;end;end;/主程序beginread_data;knapsack_order;writeln(sm);end.如果想知道选择了哪些物品,那么应将程序作些改动,具体就是对选中的物品做一标记。见参考程序,其中的数据输入采用文件输入,输入文件为bbinput.txt (第1行为背包负重能力和物品种数,第2行为每种物品的价值,第三行为每种物品的重量)。

展开阅读全文
相关资源
相关搜索

当前位置:首页 > 学术论文 > 其它学术论文

电脑版 |金锄头文库版权所有
经营许可证:蜀ICP备13022795号 | 川公网安备 51140202000112号