模板
详细解析🔗
完全背包和0-1背包的区分就是物品是否可以被多次选择,再0-1背包的滚动数组中讲过,从后向前遍历背包就是为了防止多次选择物品,所以完全背包就是从前先后选择物品。
先遍历物品:
public static void main(String[] args) {int[] weight = {1, 3, 4};int[] value = {15, 20, 30};int bagsize = 4;testweightbagproblem(weight, value, bagsize);}public static void testweightbagproblem(int[] weight, int[] value, int bagsize) {int[] dp = new int[bagsize + 1];for (int i = 0; i < weight.length; i++) {for (int j = weight[i]; j <= bagsize; j++) {dp[j] = Math.max(dp[j], dp[j - weight[i]] + value[i]);}}// 打印dp数组}
注意物品和背包的遍历可以交换遍历。
先遍历背包:(注意,先遍历背包需要判断背包能否装下商品,或者在物品for循环上做限制)
for (int i = 1; i <= bagWeight; i++){ // 遍历背包容量for (int j = 0; j < weight.length; j++){ // 遍历物品if (i - weight[j] >= 0){dp[i] = Math.max(dp[i], dp[i - weight[j]] + value[j]);}}}
在完全背包中,对于一维dp数组来说,其实两个for循环嵌套顺序同样无所谓!
因为dp[j] 是根据 下标j之前所对应的dp[j]计算出来的。 只要保证下标j之前的dp[j]都是经过计算的就可以了。
遍历物品在外层循环,遍历背包容量在内层循环,状态如图:
遍历背包容量在外层循环,遍历物品在内层循环,状态如图:
而0-1背包会存在覆盖的问题。
 
