用Python解决挖黄金矿问题

共 10935字,需浏览 22分钟

 ·

2024-03-30 08:30

0b5644f5ada926253d9c935a76ea39a9.webp

62ad2b9646d60d48fedd6b7b2833b0e3.webp

 

  问题描述

5 座金矿,每座金矿黄金储量不同,需要参与挖掘的工人数目也不同。假定有 10 个工人,每座金矿要么全挖,要么不挖,不可以拆分,请问要想得到最多的黄金,应该选择挖取哪几座金矿?金矿的信息如表 1 所示。

 1   金矿信息

金 矿 编 号

所需工人数量

黄金存储量

1

5人

400kg

2

5人

500kg

3

3人

200kg

4

4人

300kg

5

3人

350kg

 

解析问题

拿到这样一个题目,你会想到用什么算法解决呢?仔细阅读一下,有没有和背包问题很像呢?背包问题用的是动态规划算法,那么,这个问题我们试着用动态规划算法。首先我们将挖矿问题创建一个网格,如图 9 所示。

32193983ed0fde646cbb1d5d31eb9f60.webp

9   挖矿网格

从图 9 可以看出,网格的各行代表挖矿的价值以及需要的人数,各列是工人数(工人拆分成 1~10 人),网格最初是空的,等我们填满网格,就能找到答案了。

400kg/5

我们先来看 400kg/5 这行,忽略下面的 500kg/5 200kg/3 300kg/4 350kg/3 行,如图 10 所示。

b4667bb67f14c648926402bf5675fefb.webp

10  400kg/5

先来看第 1 个单元格, 1 个工人不能挖 400kg 的金矿,因此第一个单元格的存储量是 0kg ,如图 11 所示。

d343b503807d33205f80a632c3b6232f.webp

11   填第 1 个单元格

根据问题的要求,需要 5 个工人才能挖 400kg 的金矿,因此单元格 2~4 都不能挖 400kg 的金矿, 2~4 的单元格的存储量都是 0kg ,如图 12 所示。

1ca84081e2498a3a97cd0a64ee366f8c.webp

12   填第 2~4 个单元格

到第 5 个单元格,有足够的工人可以挖 400kg 的金矿了,因此第 5 个单元格的存储量是 400kg ,如图 13 所示。

46226030c4652ac1451edc4034c635a9.webp

13   填第 5 个单元格

只要超过 5 人,都能挖 400kg 的金矿,只不过剩余的工人就可以休息了。因此第 6~10 单元格的存储量也是 400kg ,如图 14 所示。

389dafa2cdfd479e4740eb1f03a80b86.webp

14   填第 6~10 个单元格

14 已经将 400kg/5 行网格填满,这行表示的最大值是 400 ,也就是说如果有 10 个工人,可挖金矿最大的黄金存储量是 400kg 。显然这不是最优解,接下来接着讲解。

500kg/5

我们再来看 500kg/5 这行,忽略下面的 200kg/3 300kg/4 350kg/3 行,如图 15 所示。

3c0f084b29f0d604ed1a4d9e5c01db90.webp

15  500kg/5

先来看第 1~4 个单元格, 1~4 个工人都不能挖 500kg 的金矿,因此 500kg/5 行第 1~4 个单元格为 0kg ,如图 16 所示。

0e9c337a32c5eaa869e994690e679b43.webp

16   填第 1~4 个单元格

再来填第 5~9 个单元格,有 5 个工人就能挖 500kg 的金矿,从 400kg/5 行可知, 5~9 个工人的可挖存储量是 400kg ,而 500 大于 400 ,因此,数据进行更新, 5~9 个工人可挖存储量是 500kg ,如 17 所示。

49d0e844293f153395a6985182ca35fc.webp

17   填第 5~9 个单元格

再来看一下第 10 个单元格, 5 个工人可以挖 400kg 的金矿, 5 个工人可以挖 500kg 的金矿,那么 10 个工人就可以分工同时挖 400kg 500kg 的金矿,因此当有 10 个工人时,可挖 400+500=900kg 的黄金存储量,如图 18 所示。

b234ef54dffe169c0180387f8d037d89.webp

18   10 个单元格

从图 18 可以看出,已经将 500kg/5 行填满,此时最大的可挖黄金存储量更新了,变成了 900 。我们再接着看下一行网格的情况。

200kg/3

我们再来看 200kg/3 这行,忽略下面的 300kg/4 350kg/3 行,如图 19 所示。

770d65d13cc4abb95d6d93d4dd1cde60.webp

19  200kg/3

根据问题含义,可知 200kg 的金矿需要 3 个工人,因此第 1 2 个单元格的可挖黄金存储量依然是 0kg ,如图 20 所示。

6a585725b5f8db582d7f6634511835ee.webp

20   填第 1 2 个单元格

当有 3 个工人时,就可以挖 200kg 的金矿了,之前 500kg/5 行所求 3 个工人可挖黄金 0kg 200 0 大,因此第 3 个单元格存储量更新了,如图 21 所示。

7239dcdf834cee895dcc25420a7fef87.webp

21   填第 3 个单元格

当有 4 个工人时,也只能挖 200kg 的金矿,因此第 4 个单元格的最大黄金存储量是 200kg ,如 22 所示。

487fab6db4734d947f2b365c7710c50b.webp

22   填第 4 个单元格

当有 5~7 个工人时,能挖 200kg 的金矿,但是在 500kg/5 行的 5~7 个工人可挖的最大黄金存储量是 500kg 500 大于 200 ,因此第 5~7 个单元格的黄金存储量依然是 500kg ,如图 23 所示。

4adce567fcc0d795989cf648820b55de.webp

23   填第 5~7 个单元格

再来看当有 8~9 个工人时的情况,可以分 3 人挖 200kg 的金矿,剩下的人可以挖 500kg 的金矿,这样算来,可以挖 200+500=700kg 的金矿, 500kg/5 行的 8~9 单元格是 500 ,很明显, 700 大于 500 ,因此数据进行更新,如图 24 所示。

3ebc3e7e3b49448a1df8be25962ae231.webp

24   填第 8~9 个单元格

再来看 , 当有 10 个工人时,从 500kg/5 行求得的最大可挖存储量是 900kg ,如果分 3 人挖 200kg 的金矿,剩余 7 人只能挖最大存储量是 500 的金矿,相加之后是 700 700 小于 900 ,因此第 10 个单元格依然是 500kg/5 行求得的 900 ,如图 25 所示。

1e98ea5536e9605291907b719bb735a0.webp

25   填第 10 个单元格

从图 25 可以看出,最大可挖黄金存储量是 900kg ,接着寻找 300kg/4 行。

300kg/4

我们再来看 300kg/4 这行,忽略下面的 350kg/3 行,如图 26 所示。

9dc9c8c48e493938bbc296bdb76ffa1f.webp

26  350kg/3

当有 1~2 个工人时,不能挖 300kg 的金矿,因此第 1 2 个单元格的值 0 ,当有 3 工人时,也不能挖 300kg 的金矿,因此第 3 个单元格的最大可挖存储量是 200kg/3 行的值 200 ,如 27 所示。

8433befa309e74189e3dd64bf1d1183b.webp

27   填第 1~3 个单元格

200kg/3 行可以看出当前第 4 个单元格的最大值是 200 ,可是当有 4 个工人时,就可以挖 300kg 的金矿了, 300 200 大,因此更新第 4 个单元格的值,如图 28 所示。

96d4cc006376bcdac2b6cb28cea459f9.webp

28   更新第 4 个单元格

当有 5~7 个工人时,可以挖 300kg 的金矿,但是很明显, 300 小于 500 ,因此第 5~7 个单元格依然是 500 ;当有 8 个工人时,能分 4 人挖 300kg 的金矿,剩余的 4 人只能挖 200kg 的金矿, 300+200=500 ,而从 200kg/3 行可知当前最大值是 700 ,很明显 700 大于 500 ,因此第 8 个单元格依然是 700 ,如图 29 所示。

cfed5c5be80226c35f7d6c22109059ad.webp

29   填第 5~8 个单元格

当有 9 个工人时,可以分 4 人挖 300kg 的金矿,分 5 人挖 500kg 的金矿,此时总共可以挖 300+500=800kg 的金矿, 800 700 大,因此更新数据,如图 30 所示。

cb4991159d41756a243dfe7876ceee6a.webp

30   填第 9 个单元格

当有 10 个工人时,之前计算的最大值是 900 10 个工人可以分 4 个工人挖 300kg ,剩下的 6 人最多可挖 500kg 的金矿,加起来是 800kg 800 小于 900 ,因此第 10 个单元格依然是 900 ,如图 31 所示。

5bfca2860412e4eb745938d453460dfa.webp

31   填第 10 个单元格

从图 31 可以看出,此时的最大可挖黄金存储量是依然是 900kg ,接下来看 350kg/3 行。有没有可能更新最大值。

350kg/3

我们再来看 350kg/3 行,如图 32 所示。

eebd4b6fbad56ee3276efde277f06a81.webp

32  350kg/3

当有 1~2 个工人时,不能挖 350kg 的金矿,因此最大可挖黄金存储量依然是 0kg ,而当有 3 个工人时,可以挖 350kg 的金矿, 350 大于 200 ,更新数据;当有 4 个工人时,也可以挖 350kg 金矿, 350 大于 300 ,更新数据,如图 33 所示。

d596fca125051b8e6141b0c3eee5faef.webp

33   填第 1~4 个单元格

当有 5 个工人时,可以挖 350kg 的金矿,但是 500 350 大,因此第 5 个单元格依然是 500 。而当有 6 个工人时,可以分 3 个人挖 350kg 的金矿,剩余 3 个人挖 200kg 的金矿,因此是 350+200=550kg 的金矿, 550 500 大,因此更新数据,如图 34 所示。

8a22da8ac17cc904b820fabffc86bf5e.webp

34   填第 5 6 个单元格

当有 7 个工人时,可以分 3 人挖 350kg 的金矿,分 4 人挖 300kg 的金矿,一共可以挖 350+300=650kg 的金矿, 650 大于 500 ,更新数据;当有 8 个工人时,可以分 3 个工人挖 350kg 的金矿,分 5 个工人挖 500kg 的金矿,一共可以挖 350+500=850kg 850 大于 700 ,更新数据,如图 35 所示。

6175c28d5ffe539388471d051d804871.webp

35   填第 7 8 个单元格

当有 9 个工人时,可以分 3 个工人挖 350kg 的金矿,剩余人挖 500kg 的金矿,一共可以挖 350+500=850kg 的金矿, 850 大于 800 ,更新数据;当有 10 个工人时,拆分工人挖矿之后的最大值是 850 850 小于 900 ,因此第 10 个单元格依然是 900 ,如图 36 所示。

1626258668868f5ad8255c4f3f51edcc.webp

36   填第 9 10 个单元格

从图 36 可以看出,此时已经填满了挖矿网格,也发现 10 个工人最大可挖黄金存储量是 900kg ,挖的金矿是 400kg/5 500kg/5 这两座金矿。

3   代码实现

从第 14.6.2 节中的分析来看,可以用动态规划算法来实现挖黄金矿问题,从图 14.10~ 14.36 可知,我们每填一个单元格,使用的公式都是相同的,公式如图 37 所示。

b1cc554e7be2645a6cc537a96bb1a7c5.webp

37   公式

我们可以利用图 37 所示的公式来填写每个单元格,最终的单元格和图 36 一致。

实例 8  挖黄金矿问题                    

具体代码如下:

"""

功能:自定义挖黄金矿函数,实现动态规划算法

参数说明: count :表示矿的数量

          TotalWorkers: 表示总的工人数

          workers :表示每个矿需要的工人数

          gold :表示每个金矿黄金储量

"""

def  mine (count, TotalWorkers, workers, gold):

置零,表示初始状态

reserves = [[0  for  in  range (TotalWorkers + 1)]  for  in  range (count +1)]

     for  in  range (1, count+1):                    遍历物每座矿,从第 1 座矿计算

         for  in  range (1, TotalWorkers + 1):           遍历工人数

             reserves[i][j] = reserves[i - 1][j]        定义 reserves 数组存储黄金的最大储存量

              有足够工人挖矿,遍历前一个状态考虑是否置换

             if  j >= workers[i - 1]  and  reserves[i][j] < reserves[i - 1][j - workers[i - 1]] + gold[i - 1]:

                最大存储量就是:当前存储量 + 剩余工人可挖存储量

              reserves[i][j] = reserves[i - 1][j - workers[i - 1]] +gold[i - 1]

     for  in  reserves:                           遍历输出挖矿网格

         print (x)

     return  reserves                           返回最大黄金的储存量

"""

功能:自定义显示输出结果函数

参数说明: count :表示矿的数量

          TotalWorkers: 表示总的工人数

          workers :表示每个矿需要的工人数

          reserves :表示最大价值,即所求的结果

"""

def  show (count, TotalWorkers, workers, reserves):

    x = [ False for  in  range (count)]                      初始化 x ,使得 x 为假

     j = TotalWorkers                             工人总人数赋给变量 j

     for  in  range (count, 0, -1):                    遍历每座矿

          如果 reserves[i][j] 单元格大于上一行同列的单元格的值,进行更新

         if  reserves[i][j] > reserves[i - 1][j]:       

             x[i - 1] =  True

             j -= workers[i - 1]                        工人数减去上一行同列的单元格使用的工人数

     print ( ' 可挖最大存储量为 :' , reserves[count][TotalWorkers])  输出最大存储量的值

     print ( ' 要挖的金矿为 :' )

     for  in  range (count):     

       if  x[i]: 

         print ( ' ' , i + 1,  ' 个矿  ' end = '' ) 

 

 

count = 5                                    一共有 5 座矿

TotalWorkers = 10                              一共有 10 个工人

workers = [5,5,3,4,3]                            每个矿需要的工人数

gold = [400,500,200,300,350]                   每个矿的存储黄金量

reserves = mine(count, TotalWorkers, workers, gold)   调用 mine() 函数动态规划算法函数

show(count, TotalWorkers, workers, reserves)        调用 show() 函数输出结果

程序运行结果如图 38 所示。

48dbdde86a121db5d2e8d315245e80fb.webp

38   挖黄金矿问题

从图 38 的结果来看,和图 36 分析的结果一致。最后输出的所挖的金矿是第 1 个和第 2 个,即 400kg/5 500kg/5

9dc1367ae1027b76c55415689e6ae9c5.webp

浏览 9
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报
评论
图片
表情
推荐
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报