首页 > 基础资料 博客日记
金矿问题
2025-01-12 22:30:03基础资料围观27次
10.金矿问题
题目
有5座金矿,每座金矿的黄金储量不同,需要参与挖掘的工人人数也不同。例如有的金矿储量是500kg黄金,需要5个工人来挖掘,有的金矿储量是200kg黄金,需要3个工人来挖掘......
如果参与挖矿的工人的总数是10。每座金矿要么全挖,要么不挖,不能派出一半人挖取一半的金矿。要求计算出,最大挖取的黄金值。
int[] g = {400, 500, 200, 300, 350};//金矿储量
int[] p = {5, 5, 3, 4, 3};//挖取需要的工人数量
思路
如果使用贪心算法,局部情况最优解,但是整体上却未必最优。比如,计算每个金矿的人均产值,为{80,100,66.6,75,116.6},单位是kg/人,每一步最优的话,就是挖取350/3人的金矿,和500/5人的金矿,然后就无法挖取了,这里挖取的金矿是850kg,但是还有比这个高的,400/5,500/5人的挖取就是900kg。
这里提供正确思路,其实这个问题,是典型的动态规划
题目。动态规划就是把复杂的问题简化为规模较小的子问题,再从简单的子问题自底向上的一步一步递推,最终得到复杂问题的最优解。
对于金矿,只有挖和不挖,两种选择。假设350kg/3人的,金矿不挖,问题就会转成10人挖前4个金矿的最优解,但是这个只是假设的最优,是否最优,还需要和挖350kg/3人的金矿,然后加上7个人挖前4个的金矿的最优解。比较那个大,那个就是最优解。这两种情况,就是全局问题的最优子结构
。
就这样,对于每个金矿,都有挖和不挖,问题就一分为二,二分为四,一直把问题简化到0个金矿或0个工人时的最优选择,这个收益结果显然是0,也就是问题的边界
。
这些就是动态规划的要点,确定问题的,全局最优解,和最优子结构之间的关系,以及问题的边界。这个关系用数学公式表达的话,就是状态转移方程式
。
金矿数量为n,工人数量为w,金矿的含金量设为数组g[],金矿所需开采人数为p[]。设f(n,w)为n个金矿,w个工人时的最优收益函数,则该问题的状态转移方程式
为,特别重要,后续2,3版本的代码,都要理解这个状态转移函数。
f(n,w) = 0(n=0 或 w=0) //问题的边界
f(n,w) = f(n-1,w) (n>=1,w<p[n-1]) //工人数量,不够挖取当前金矿
f(n,w) = max(f(n-1,w),f(n-1,w-p[n-1]) + g[n-1]) (n >=1,w>=p[n-1]) //工人数量,可以挖取当前金矿
代码
//递归
/**
* 获取金矿最佳收益
*
* @param n 当前金矿数量
* @param w 当前工人人数
* @param g 金矿含量数组
* @param p 对应金矿含量所需人数数组
* @return 返回当前工人人数,和金矿数量的最佳收益函数
* 递归方式计算,的时间复杂度为O(2^n),适合于金矿数量小时的情况
* 注意这个n是当前金矿的数量,就好比每一个金矿都要分成2种情况,每个金矿下的其他金矿又有两种情况,组合起来看是一个满二叉树,
* 节点的个数和层级有关也就是金矿的数量有关为2^n(n为层级,这个问题中为金矿的个数)
*/
public static int getBestGoldMining(int n, int w, int[] g, int[] p) {
if (n <= 0 || w <= 0)
return 0;
if (w < p[n - 1])
return getBestGoldMining(n - 1, w, g, p);
return Math.max(getBestGoldMining(n - 1, w, g, p), getBestGoldMining(n - 1, w - p[n - 1], g, p) + g[n - 1]);
}
注意,上述代码是正确的,但是时间复杂度,却来到了2^n。低效的原因,是进行了很多重复运算。
可以,使用一个表格,二维数组,来存储这些中间值。表格最左侧代表不同的金矿选择范围,从上到下,每多增加1行,就代表多1个金矿可供选择,也就是f(n,w)函数中的n值。此时,最后1行最后1个格子的值,就是最终要求的结果,即5个金矿、10个工人的最优收益。
1个工人 | 2个工人 | 3个工人 | 4个工人 | 5个工人 | 6个工人 | 7个工人 | 8个工人 | 9个工人 | 10个工人 | |
---|---|---|---|---|---|---|---|---|---|---|
400kg/5人 | ||||||||||
500kg5人 | ||||||||||
200kg/3人 | f(3,5) | |||||||||
300kg/4人 | ||||||||||
350kg/3人 | f(5,10) |
/**
* @param n 金矿数量 这个值,是等于g.length的,可以省略
* @param w 当前工人数量
* @param g 每个金矿存储数量数组
* @param p 每个金矿开采所需工人数量数组
* @return 时间复杂杜和空间复杂度都为O(nw) 在w值较小的情况下,该算法比递归的性能好,如果w很大的,而n很小的情况下,
* 就是2^n < nw的情况下,递归会比该算法好。
* 算法是对不同情况下问题解决的一种方式,没有任何情况下都最优的解法,只有适合当前情况的不错的选择。
*/
public static int getBestGoldMiningV2(int n, int w, int[] g, int[] p) {
//抽象一个二维数组,用于暂存最优收益函数的临时结果
int[][] result = new int[n + 1][w + 1];
/*
* 为什么是n+1和w+1,因为要利用int[][]数组的默认值为0这个特性,以及在当前问题下,n=0,和w=0时,f(n,w) = 0
* int[0][x](0=<x and x<= w)的情况都为0,int[x][0](0=<x and x<=n)的情况都为0,用其作为初始值,计算f(1,x)(1=<x and x <=w)的值,
* 也就是表格中第一行的值,再用第一行的值,计算第二行,实际上只用了,这个二维数组的一部分存储中间数据result[1][1] ~ result[n][w]
*/
//i从1开始
for (int i = 1; i <= n; i++) {
//j从1开始
for (int j = 1; j <= w; j++) {
if(j < p[i-1])
result[i][j] = result[i-1][j];//第一行
else
result[i][j] = Math.max(result[i-1][j],result[i-1][j-p[i-1]] + g[i-1]);
}
}
return result[n][w];
}
但是,上述代码还不够最优,是因为每一行的值,都是由上一行计算得来的,我们要的只是最后一行的最后一列的值,所以代码还可以简化。
/**
* @param n 金矿数量 这个值,是等于g.length的,可以省略
* @param w 当前工人数量
* @param g 每个金矿存储数量数组
* @param p 每个金矿开采所需工人数量数组
* @return 时间复杂度O(nw),空间复杂杜为0(n)
*/
public static int getBestGoldMiningV3(int n, int w, int[] g, int[] p){
int[] results = new int[w+1];//暂存一行的临时数据
//填充一位数组
for(int i = 1;i <= g.length; i++){
//从右往左,填充数据
for(int j = w; j >= 1; j--){
if(j >= p[i-1]){
results[j] = Math.max(results[j], results[j-p[i-1]] + g[i-1]);
}
}
}
return results[w];
}
只是为了记录自己的学习历程,且本人水平有限,不对之处,请指正。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:jacktools123@163.com进行投诉反馈,一经查实,立即删除!
标签: