有n件物品和一个最多能背重量为w 的背包。第i件物品的重量是weight[i],得到的价值是value[i] 。每件物品只能用一次,求解将哪些物品装入背包里物品价值总和最大。

dp分析:

  1. 确定dp数组以及下标的含义

使用二维数组,即dp[i][j] 表示从下标为[0-i]的物品里任意取,放进容量为j的背包,价值总和最大是多少

  1. 确定递推公式

回顾一下dp[i][j]的含义:从下标为[0-i]的物品里任意取,放进容量为j的背包,价值总和最大是多少。 那么可以有两个方向推出来dp[i][j],

  • 不放物品i:由dp[i - 1][j]推出,即背包容量为j,里面不放物品i的最大价值,此时dp[i][j]就是dp[i - 1][j]。(其实就是当物品i的重量大于背包j的重量时,物品i无法放进背包中,所以被背包内的价值依然和前面相同。)
  • 放物品i:由dp[i - 1][j - weight[i]]推出,dp[i - 1][j - weight[i]] 为背包容量为j - weight[i]的时候不放物品i的最大价值,那么dp[i - 1][j - weight[i]] + value[i] (物品i的价值),就是背包放物品i得到的最大价值

所以递归公式: dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - weight[i]] + value[i]);

  1. dp数组如何初始化
  2. 确定遍历顺序

先遍历物品

  1. // weight数组的大小 就是物品个数
  2. for(int i = 1; i < weight.size(); i++) { // 遍历物品
  3. for(int j = 0; j <= bagweight; j++) { // 遍历背包容量
  4. if (j < weight[i]) dp[i][j] = dp[i - 1][j];
  5. else dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - weight[i]] + value[i]);
  6. }
  7. }
  1. 举例推导dp数组
  1. public static void main(string[] args) {
  2. int[] weight = {1, 3, 4};
  3. int[] value = {15, 20, 30};
  4. int bagsize = 4;
  5. testweightbagproblem(weight, value, bagsize);
  6. }
  7. public static void testweightbagproblem(int[] weight, int[] value, int bagsize){
  8. int wlen = weight.length, value0 = 0;
  9. //定义dp数组:dp[i][j]表示背包容量为j时,前i个物品能获得的最大价值
  10. int[][] dp = new int[wlen + 1][bagsize + 1];
  11. //初始化:背包容量为0时,能获得的价值都为0
  12. for (int i = 0; i <= wlen; i++){
  13. dp[i][0] = value0;
  14. }
  15. //遍历顺序:先遍历物品,再遍历背包容量
  16. for (int i = 1; i <= wlen; i++){
  17. for (int j = 1; j <= bagsize; j++){
  18. if (j < weight[i - 1]){
  19. dp[i][j] = dp[i - 1][j];
  20. }else{
  21. dp[i][j] = math.max(dp[i - 1][j], dp[i - 1][j - weight[i - 1]] + value[i - 1]);
  22. }
  23. }
  24. }
  25. //打印dp数组
  26. for (int i = 0; i <= wlen; i++){
  27. for (int j = 0; j <= bagsize; j++){
  28. system.out.print(dp[i][j] + " ");
  29. }
  30. system.out.print("\n");
  31. }
  32. }

简化版本:
dp[j]表示:容量为j的背包,所背的物品价值可以最大为dp[j],那么dp[0]就应该是0,因为背包容量为0所背的物品的最大价值就是0。
倒序遍历是为了保证物品i只被放入一次!。但如果一旦正序遍历了,那么物品0就会被重复加入多次!

  1. public static void main(String[] args) {
  2. int[] weight = {1, 3, 4};
  3. int[] value = {15, 20, 30};
  4. int bagWight = 4;
  5. testWeightBagProblem(weight, value, bagWight);
  6. }
  7. public static void testWeightBagProblem(int[] weight, int[] value, int bagWeight){
  8. int wLen = weight.length;
  9. //定义dp数组:dp[j]表示背包容量为j时,能获得的最大价值
  10. int[] dp = new int[bagWeight + 1];
  11. //遍历顺序:先遍历物品,再遍历背包容量
  12. for (int i = 0; i < wLen; i++){
  13. for (int j = bagWeight; j >= weight[i]; j--){
  14. dp[j] = Math.max(dp[j], dp[j - weight[i]] + value[i]);
  15. }
  16. }
  17. //打印dp数组
  18. for (int j = 0; j <= bagWeight; j++){
  19. System.out.print(dp[j] + " ");
  20. }
  21. }

416. 分割等和子集

image.png

解题思路: 类似于背包问题,可进行转化。

  • 背包的体积为sum / 2
  • 背包要放入的商品(集合里的元素)重量为 元素的数值,价值也为元素的数值
  • 背包如果正好装满,说明找到了总和为 sum / 2 的子集。
  • 背包中每一个元素是不可重复放入。
  1. public boolean canPartition(int[] nums) {
  2. int total = 0;
  3. for (int num : nums) {
  4. total += num;
  5. }
  6. //定义dp数组:dp[i][j]表示背包容量为total/2时,前i个数能获得的最大和
  7. if (total % 2 != 0) {
  8. return false;
  9. }
  10. int[] dp = new int[total / 2 + 1];
  11. for (int i = 1; i < nums.length; i++) {
  12. for (int j = total / 2; j >= nums[i]; j--) {
  13. dp[j] = Math.max(dp[j], dp[j - nums[i]] + nums[i]);
  14. }
  15. }
  16. return dp[total / 2] == total / 2;
  17. }

1049. 最后一块石头的重量 II

image.png

解题思路: 相同于上一道题,可以获得最大背包的重量为 (sum / 2) ,然后转换为背包问题,重量总和 - 2 (sum / 2) , 就是*最小的可能重量。 本题其实就是尽量让石头分成重量相同的两堆,相撞之后剩下的石头最小

 //背包问题, 求两堆石头的最小差值
    public static int lastStoneWeightII(int[] stones) {
        int total = 0;
        for (int stone : stones) {
            total += stone;
        }
        int half = total / 2;
        int[] dp = new int[half + 1];
        //遍历顺序:先遍历物品,再遍历背包容量
        for (int i = 0; i < stones.length; i++) {
            for (int j = half; j >= stones[i]; j--) {
                dp[j] = Math.max(dp[j], dp[j - stones[i]] + stones[i]);
            }
        }

        return total - dp[half] * 2;

    }