对于动态规划 c++

上传人:野鹰 文档编号:12689748 上传时间:2017-10-20 格式:DOC 页数:13 大小:71KB
返回 下载 相关 举报
对于动态规划 c++_第1页
第1页 / 共13页
对于动态规划 c++_第2页
第2页 / 共13页
对于动态规划 c++_第3页
第3页 / 共13页
对于动态规划 c++_第4页
第4页 / 共13页
对于动态规划 c++_第5页
第5页 / 共13页
点击查看更多>>
资源描述

《对于动态规划 c++》由会员分享,可在线阅读,更多相关《对于动态规划 c++(13页珍藏版)》请在金锄头文库上搜索。

1、 对于动态规划,每个刚接触的人都需要一段时间来理解,特别是第一次接触的时候总是想不通为什么这种方法可行,这篇文章就是为了帮助大家理解动态规划,并通过讲解基本的 01 背包问题来引导读者如何去思考动态规划。本文力求通俗易懂,无异性,不让读者感到迷惑,引导读者去思考,所以如果你在阅读中发现有不通顺的地方,让你产生错误理解的地方,让你难得读懂的地方,请跟贴指出,谢谢! -第一节-初识动态规划-经典的 01 背包问题是这样的:有一个包和 n 个物品,包的容量为 m,每个物品都有各自的体积和价值,问当从这 n个物品中选择多个物品放在包里而物品体积总数不超过包的容量 m 时,能够得到的最大价值是多少?对于

2、每个物品不可以取多次,最多只能取一次,之所以叫做 01 背包,0 表示不取,1 表示取为了用一种生动又更形象的方式来讲解此题,我把此题用另一种方式来描述,如下:有一个国家,所有的国民都非常老实憨厚,某天他们在自己的国家发现了十座金矿,并且这十座金矿在地图上排成一条直线,国王知道这个消息后非常高兴,他希望能够把这些金子都挖出来造福国民,首先他把这些金矿按照在地图上的位置从西至东进行编号,依次为 0、1、2、 3、4、5 、6 、7、8、9,然后他命令他的手下去对每一座金矿进行勘测,以便知道挖取每一座金矿需要多少人力以及每座金矿能够挖出多少金子,然后动员国民都来挖金子。题目补充 1:挖每一座金矿需

3、要的人数是固定的,多一个人少一个人都不行。国王知道每个金矿各需要多少人手,金矿 i 需要的人数为 peopleNeeded。题目补充 2:每一座金矿所挖出来的金子数是固定的,当第 i 座金矿有 peopleNeeded人去挖的话,就一定能恰好挖出 gold 个金子。否则一个金子都挖不出来。题目补充 3:开采一座金矿的人完成开采工作后,他们不会再次去开采其它金矿,因此一个人最多只能使用一次。题目补充 4:国王在全国范围内仅招募到了 10000 名愿意为了国家去挖金子的人,因此这些人可能不够把所有的金子都挖出来,但是国王希望挖到的金子越多越好。题目补充 5:这个国家的每一个人都很老实(包括国王)

4、,不会私吞任何金子,也不会弄虚作假,不会说谎话。题目补充 6:有很多人拿到这个题后的第一反应就是对每一个金矿求出平均每个人能挖出多少金子,然后从高到低进行选择,这里要强调这种方法是错的,如果你也是这样想的,请考虑背包模型,当有一个背包的容量为 10,共有 3 个物品,体积分别是 3、3、5,价值分别是 6、6 、9 ,那么你的方法取到的是前两个物品,总价值是 12,但明显最大值是后两个物品组成的 15。题目补充 7:我们只需要知道最多可以挖出多少金子即可,而不用关心哪些金矿挖哪些金矿不挖。那么,国王究竟如何知道在只有 10000 个人的情况下最多能挖出多少金子呢?国王是如何思考这个问题的呢?

5、国王首先来到了第 9 个金矿的所在地(注意,第 9 个就是最后一个,因为是从 0 开始编号的,最西边的那个金矿是第 0 个) ,他的臣子告诉他,如果要挖取第 9 个金矿的话就需要 1500 个人,并且第 9 个金矿可以挖出 8888 个金子。听到这里国王哈哈大笑起来,因为原先他以为要知道十个金矿在仅有 10000 个人的情况下最多能挖出多少金子是一件很难思考的问题,但是,就在刚才听完他的臣子所说的那句话时,国王已经知道总共最多能挖出多少金子了,国王是如何在不了解其它金矿的情况下知道最多能挖出多少金子的呢?他的臣子们也不知道这个谜,因此他的臣子们就问他了:“最聪明的国王陛下,我们都没有告诉您其它

6、金矿的情况,您是如何知道最终答案的呢?”得意的国王笑了笑,然后把他最得意的“左、右手” 叫到跟前,说到:“我并不需要考虑最终要挖哪些金矿才能得到最多的金子,我只需要考虑我面前的这座金矿就可以了,对于我面前的这座金矿不外乎仅有两种选择,要么挖,要么不挖,对吧?”“当然,当然”大臣们回答倒。国王继续说道:“如果我挖取第 9 座金矿的话那么我现在就能获得 8888 个金子,而我将用去 1500 个人,那么我还剩下 8500 个人。我亲爱的左部下,如果你告诉我当我把所有剩下的 8500 个人和所有剩下的其它金矿都交给你去开采你最多能给我挖出多少金子的话,那么我不就知道了在第 9 个金矿一定开采的情况下

7、所能得到的最大金币数吗?”国王的左部下听后回答道:“国王陛下,您的意思是如果我能用 8500 个人在其它金矿最多开采出 x 个金币的话,那您一共就能够获得 x + 8888 个金子,对吗?”“是啊,是啊如果第 9 座金矿一定开采的话”大臣们点头说到。国王笑着继续对着他的右部下说到:“亲爱的右部下,也许我并不打算开采这第 9 座金矿,那么我依然拥有 10000 个人,如果我把这 10000 个人和剩下的金矿都给你的话,你最多能给我挖出多少个金子呢?”国王的右部下聪明地说道:“尊敬的国王陛下,我明白您的意思了,如果我回答最多能购开采出 y 个金币的话,那您就可以在 y 和 x+8888 之间选择一

8、个较大者,而这个较大者就是最终我们能获得的最大金币数,您看我这样理解对吗?”国王笑得更灿烂了,问他的左部下:“那么亲爱的左部下,我给你 8500 个人和其余金矿的话你能告诉我最多能挖出多少金子吗?”“请您放心,这个问题难不倒我”。左部下向国王打包票说到。国王高兴地继续问他的右部下:“那右部下你呢,如果我给你 10000 个人和其余金矿的话你能告诉我最多能挖出多少金子吗?”“当然能了!交给我吧!”右部下同左部下一样自信地回答道。 “那就拜托给你们两位了,现在我要回到我那舒适的王宫里去享受了,我期待着你们的答复。 ”国王说完就开始动身回去等消息了,他是多么地相信他的两个大臣能够给他一个准确的答复,

9、因为国王其实知道他的两位大臣要比他聪明得多。故事发展到这里,你是否在想国王的这两个大臣又是如何找到让国王满意的答案的呢?他们为什么能够如此自信呢?事实上他们的确比国王要聪明一些,因为他们从国王的身上学到了一点,就是这一点让他们充满了自信。国王走后,国王的左、右部下来到了第 8 座金矿,早已在那里等待他们的金矿勘测兵向两位大臣报道:“聪明的两位大臣,您们好,第 8 座金矿需要 1000 个人才能开采,可以获得 7000 个金子”。因为国王仅给他的左部下 8500 个人,所以国王的左部下叫来了两个人,对着其中一个人问到:“如果我给你 7500 个人和除了第 8、第 9 的其它所有金矿的话,你能告诉

10、我你最多能挖出多少金子吗?”然后国王的左部下继续问另一个人:“如果我给你 8500 个人和除了第 8、第 9 的其它所有金矿的话,你能告诉我你最多能挖出多少金子吗?”国王的左部下在心里想着:“如果他们俩都能回答我的问题的话,那国王交给我的问题不就解决了吗?哈哈哈!”因为国王给了他的右部下 10000 个人,所以国王的右部下同样也叫来了两个人,对着其中一个人问:“如果我给你 9000 个人和除了第 8、第 9 的其它所有金矿的话,你能告诉我你最多能挖出多少金子吗?”然后国王的右部下继续问他叫来的另一个人:“如果我给你 10000 个人和除了第 8、第 9 的其它所有金矿的话,你能告诉我你最多能挖

11、出多少金子吗?”此时,国王的右部下同左部下一样,他们都在为自己如此聪明而感到满足。当然,这四个被叫来的人同样自信地回答没有问题,因为他们同样地从这两个大臣身上学到了相同的一点,而两位自认为自己一样很聪明的大臣得意地笑着回到了他们的府邸,等着别人回答他们提出来的问题,现在你知道了这两个大臣是如何解决国王交待给他们的问题了吗?那么你认为被大臣叫去的那四个人又是怎么完成大臣交给他们的问题的呢?答案当然是他们找到了另外八个人!没用多少功夫,这个问题已经在全国传开了,更多人的人找到了更更多的人来解决这个问题,而有些人却不需要去另外找两个人帮他,哪些人不需要别人的帮助就可以回答他们的问题呢?很明显,当被问

12、到给你 z 个人和仅有第 0 座金矿时最多能挖出多少金子时,就不需要别人的帮助,因为你知道,如果 z 大于等于挖取第 0 座金矿所需要的人数的话,那么挖出来的最多金子数就是第 0 座金矿能够挖出来的金子数,如果这 z 个人不够开采第 0 座金矿,那么能挖出来的最多金子数就是 0,因为这唯一的金矿不够人力去开采。让我们为这些不需要别人的帮助就可以准确地得出答案的人们鼓掌吧,这就是传说中的底层劳动人民!故事讲到这里先暂停一下,我们现在重新来分析一下这个故事,让我们对动态规划有个理性认识。子问题:国王需要根据两个大臣的答案以及第 9 座金矿的信息才能判断出最多能够开采出多少金子。为了解决自己面临的问

13、题,他需要给别人制造另外两个问题,这两个问题就是子问题。思考动态规划的第一点-最优子结构:国王相信,只要他的两个大臣能够回答出正确的答案(对于考虑能够开采出的金子数,最多的也就是最优的同时也就是正确的) ,再加上他的聪明的判断就一定能得到最终的正确答案。我们把这种子问题最优时母问题通过优化选择后一定最优的情况叫做“最优子结构” 。思考动态规划的第二点-子问题重叠:实际上国王也好,大臣也好,所有人面对的都是同样的问题,即给你一定数量的人,给你一定数量的金矿,让你求出能够开采出来的最多金子数。我们把这种母问题与子问题本质上是同一个问题的情况称为“子问题重叠” 。然而问题中出现的不同点往往就是被子问

14、题之间传递的参数,比如这里的人数和金矿数。思考动态规划的第三点-边界:想想如果不存在前面我们提到的那些底层劳动者的话这个问题能解决吗?永远都不可能!我们把这种子问题在一定时候就不再需要提出子子问题的情况叫做边界,没有边界就会出现死循环。思考动态规划的第四点-子问题独立:要知道,当国王的两个大臣在思考他们自己的问题时他们是不会关心对方是如何计算怎样开采金矿的,因为他们知道,国王只会选择两个人中的一个作为最后方案,另一个人的方案并不会得到实施,因此一个人的决定对另一个人的决定是没有影响的。我们把这种一个母问题在对子问题选择时,当前被选择的子问题两两互不影响的情况叫做“子问题独立”。这就是动态规划,

15、具有“最优子结构” 、 “子问题重叠” 、 “边界”和“ 子问题独立”,当你发现你正在思考的问题具备这四个性质的话,那么恭喜你,你基本上已经找到了动态规划的方法。有了上面的这几点,我们就可以写出动态规划的转移方程式,现在我们来写出对应这个问题的方程式,如果用 goldmineNum表示第 mineNum 个金矿能够挖出的金子数,用peopleNeededmineNum表示挖第 mineNum 个金矿需要的人数,用函数f(people,mineNum)表示当有 people 个人和编号为 0、1、2、3、mineNum 的金矿时能够得到的最大金子数的话,f(people,mineNum) 等于什

16、么呢?或者说f(people,mineNum)的转移方程是怎样的呢?答案是:当 mineNum = 0 且 people = peopleNeededmineNum时 f(people,mineNum) = goldmineNum当 mineNum = 0 且 people 4. 5. #include 6. 7. using namespace std; 8. 9. 10. 11.const int max_n = 100;/程序支持的最多金矿数 12. 13.const int max_people = 10000;/程序支持的最多人数 14. 15. 16. 17.int n;/金矿数 18. 19.int peopleTotal;/可以用于挖金子的人数 20. 21.int peopleNeedmax_n;/每座金矿需要的人数 22. 23.in

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

当前位置:首页 > 行业资料 > 其它行业文档

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