动态规划--01背包问题详解

代码随想录day42和day43 动态规划 模块01背包问题
“即使到不了远方,心中也要有远方的模样。”

1. 01背包理论基础

1.1什么是背包问题

在这里插入图片描述
  背包问题大致可以分为以上几类,背包问题也就是选取物品放入重量为n的背包中,然后求背包的最大价值。

1.2二维dp数组01背包

  01背包问题—有n件物品和一个最多能背重量为w 的背包。第i件物品的重量是weight[i],得到的价值是value[i] 。每件物品只能用一次,求解将哪些物品装入背包里物品价值总和最大。
在这里插入图片描述
例子:假设背包最大重量为4,然后有三个物品,看如何将物品放入背包中能获取最大价值。
在这里插入图片描述

还是可以用动态规划的做题步骤来分析

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

dp[i][j]对应的就是取索引[0,i]的物品,放入重量为j的背包中,得到价值dp[i][j]

   2. 确定递推公式

可以先将每种情况列举出来,然后再推导公式
在这里插入图片描述
由此可以推导出递推公式为dp[i][j]=Math.max(dp[i-1][j],dp[i-1][j-weight[i]]+value[i])
    每次遍历到dp[i][j]的时候都有两种取值情况dp[i-1][j]表示不放物品时的情况  
   dp[i-1][j-weight[i]]+value[i],weight[i]表示i的重量,value[i]表示i的价值。这就是表示添加新物品的情况
在上面两种情况中取最值

  3. dp数组的初始化问题

当背包重量为0时,可以初始化dp[i][0]=0;
在这里插入图片描述

  4.确定遍历顺序

这题的从前往后面遍历,可以先遍历背包再遍历物品也可以先便利物品再遍历背包,一般是先遍历物品

  5. 举例推导dp公式
在这里插入图片描述
代码示例

    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 wlen = weight.length, value0 = 0;
        //定义dp数组:dp[i][j]表示背包容量为j时,前i个物品能获得的最大价值
        int[][] dp = new int[wlen + 1][bagsize + 1];
        //初始化:背包容量为0时,能获得的价值都为0
        for (int i = 0; i <= wlen; i++){
            dp[i][0] = value0;
        }
        //遍历顺序:先遍历物品,再遍历背包容量
        for (int i = 1; i <= wlen; i++){
            for (int j = 1; j <= bagsize; j++){
                if (j < weight[i - 1]){
                    dp[i][j] = dp[i - 1][j];
                }else{
                    dp[i][j] = Math.max(dp[i - 1][j], dp[i - 1][j - weight[i - 1]] + value[i - 1]);
                }
            }
        }
        //打印dp数组
        for (int i = 0; i <= wlen; i++){
            for (int j = 0; j <= bagsize; j++){
                System.out.print(dp[i][j] + " ");
            }
            System.out.print("\n");
        }
    }
1.3一维dp数组(滚动数组)01背包

  上面用二维数组来解决01背包问题,其实也可以用一维数组(也叫做滚动数组)来解决01背包问题。其核心也就是将二维数组压缩成一维数组。
假设背包最大重量为4,然后有三个物品,看如何将物品放入背包中能获取最大价值。
在这里插入图片描述

  那还是按照动态规划的做题步骤来分析

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

dp[j]对应的就是背包重量为j的时候对应的最大价值dp[j]

   2. 确定递推公式

可以先将每种情况列举出来,然后再推导公式
在这里插入图片描述

dp[j]是由dp[j-weight[i]]决定的,递推公式可以如下
dp[j]=Math.max(dp[j],dp[j-weight[i]]+value[i])

  3. dp数组的初始化问题

当背包重量为0时,可以初始化dp[0]=0;
在这里插入图片描述

  4.确定遍历顺序

这题先遍历物品的时候可以从前面往后面遍历,然后再背包的时候得从后面往前面进行倒序遍历。如果是顺序遍历的话物品就会重复添加,因为每个物品只能用一次,以免影响其他结果

  5. 举例推导dp公式
在这里插入图片描述
代码示例

    public static void main(String[] args) {
        int[] weight = {1, 3, 4};
        int[] value = {15, 20, 30};
        int bagWight = 4;
        testWeightBagProblem(weight, value, bagWight);
    }

    public static void testWeightBagProblem(int[] weight, int[] value, int bagWeight){
        int wLen = weight.length;
        //定义dp数组:dp[j]表示背包容量为j时,能获得的最大价值
        int[] dp = new int[bagWeight + 1];
        //遍历顺序:先遍历物品,再遍历背包容量
        for (int i = 0; i < wLen; i++){
            for (int j = bagWeight; j >= weight[i]; j--){
                dp[j] = Math.max(dp[j], dp[j - weight[i]] + value[i]);
            }
        }
        //打印dp数组
        for (int j = 0; j <= bagWeight; j++){
            System.out.print(dp[j] + " ");
        }
    }

2.leetcode 416.分割等和子集

力扣题目链接
在这里插入图片描述

2.1 详细思路及思考难点

  这题可以将其划分为01背包问题,给定一个数组,判断是否能将数组分成两个子集和相同的两个数组。要将数组分成两个相等的子集的话,数组中的和sum也就是必须是偶数(这是第一个判断条件),然后令target=sum/2;这就是子集的和,把target想象成一个背包的最大重量,现在就是让背包里面的物品的重量刚好能达到背包的最大重量,如果能达到就返回true,反之,return false。

2.2具体步骤及代码实现

  按照动态规划的做题步骤来分析

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

dp[j]对应的就是背包重量为j的时候对应的最大价值dp[j]

   2. 确定递推公式

dp[j]是由dp[j-weight[i]]决定的,递推公式可以如下
dp[j]=Math.max(dp[j],dp[j-nums[i]]+nums[i]),这里nums[i]既可以表示重量也可以表示价值

  3. dp数组的初始化问题

当背包重量为0时,可以初始化dp[0]=0;

  4.确定遍历顺序

这题先遍历物品的时候可以从前面往后面遍历,然后再背包的时候得从后面往前面进行倒序遍历。因为每个物品只能用一次,以免影响其他结果

  5. 举例推导dp公式
在这里插入图片描述
代码实现

class Solution {
    public boolean canPartition(int[] nums) {
     if(nums==null || nums.length==0)return false;
     int sum=0;
     for(int a:nums){
        sum+=a;
     }
     if(sum%2!=0)return false;
     int targrt=sum/2;
     int[] dp=new int[targrt+1];
     //dp[0]=0;
     for(int i=0;i<nums.length;i++){
         for(int j=targrt;j>=nums[i];j--){
            dp[j]=Math.max(dp[j],dp[j-nums[i]]+nums[i]);
         }
     }
     return dp[targrt]==targrt;
    }
}

3.leetcode 1049.最后一块石头的重量

力扣题目链接
在这里插入图片描述

3.1详细思路及思考难点

  这题思路跟416.分割子集问题基本一致,可以将所有石头分成两份和相同的子集和,target=sum/2,然后将石头存入dp[target]中所能满足的最大价值,最终返回的结果是(sum-dp[target])-dp[target],(sum-dp[target])表示没有装入背包的所有的集合,然后减去装入背包的,就是碰撞之后剩下的量。

3.2具体步骤及代码实现

  按照动态规划的做题步骤来分析

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

dp[j]对应的就是背包重量为j的时候对应的最大石头量dp[j]

   2. 确定递推公式

dp[j] = max(dp[j], dp[j - stones[i]] + stones[i]);

  3. dp数组的初始化问题

当背包重量为0时,可以初始化dp[0]=0;

  4.确定遍历顺序

这题先遍历物品的时候可以从前面往后面遍历,然后再背包的时候得从后面往前面进行倒序遍历。因为每个物品只能用一次,以免影响其他结果

  5. 举例推导dp公式
在这里插入图片描述
代码实现

class Solution {
    public int lastStoneWeightII(int[] stones) {
    int sum=0;
    for(int a:stones){
     sum+=a;
    }
    int target=sum>>1;
     int[] dp=new int[target+1];
     dp[0]=0;
     for(int i=0;i<stones.length;i++){
         for(int j=target;j>=stones[i];j--){
             dp[j]=Math.max(dp[j],dp[j-stones[i]]+stones[i]);
         }
     }
     return sum-dp[target]-dp[target];
    }
}

4.leetcode 494.目标和

力扣题目链接
在这里插入图片描述

4.1详细思路及思考难点

  将nums中的数值添加正负号之后,设正数和为x,那么负数的和就是sum-x(sum是nums数组中所有元素的和)。x-(sum-x)=target ==>x=(sum+target)/2。然后将x转换成填满大小为x的背包,需要dp[j]种方法。

4.2具体步骤及代码实现

  按照动态规划的做题步骤来分析

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

dp[j]对应的就是背包重量为j的时候对应的最大价值dp[j]

   2. 确定递推公式

根据下面得出递推公式为dp[j]+=dp[j-nums[i]]

  • 已经有一个1(nums[i]) 的话,有 dp[4]种方法 凑成 dp[5]。
  • 已经有一个2(nums[i]) 的话,有 dp[3]种方法 凑成 dp[5]。
  • 已经有一个3(nums[i]) 的话,有 dp[2]中方法 凑成 dp[5]
  • 已经有一个4(nums[i]) 的话,有 dp[1]中方法 凑成 dp[5]
  • 已经有一个5 (nums[i])的话,有 dp[0]中方法 凑成 dp[5]

  3. dp数组的初始化问题

当背包重量为0时,可以初始化dp[0]=0;

  4.确定遍历顺序

这题先遍历物品的时候可以从前面往后面遍历,然后再背包的时候得从后面往前面进行倒序遍历。因为每个物品只能用一次,以免影响其他结果

代码实现

class Solution {
    public int findTargetSumWays(int[] nums, int target) {
     int sum=0;
     for(int a:nums){
        sum+=a;
     }
     int zs=(sum+target)/2;
      if((target+sum)%2!=0) return 0;
     if(zs<0)zs=-zs;
     int[] dp=new int[zs+1];
     if(target>sum)return 0;
     dp[0]=1;
     for(int i=0;i<nums.length;i++){
         for(int j=zs;j>=nums[i];j--){
             dp[j]+=dp[j-nums[i]];
         }
     }
     return dp[zs];
    }
}

5.leetcode 474.一和零

力扣题目链接
在这里插入图片描述

5.1详细思路及思考难点

  这题用二维dp数组来做,把每个字符串当作单独的物品,总共的0和1的个数当作背包的容量。

5.2具体步骤及代码实现

  按照动态规划的做题步骤来分析

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

dp[i][j]表示当有i个0和j个1时的最大子集的大小

   2. 确定递推公式

根据下面得出递推公式为dp[i][j]=Math.max(dp[i][j],dp[i-zero][j-one]+1)

  3. dp数组的初始化问题

初始为0就可以

  4.确定遍历顺序

遍历字符串的时候顺序遍历,遍历0和1的时候倒叙

  5.推导dp数组
在这里插入图片描述
代码实现

class Solution {
    public int findMaxForm(String[] strs, int m, int n) {
       int[][] dp=new int[m+1][n+1];
       int zero,one;
       for(String s : strs){
           zero=0;
           one=0;
           for(char c:s.toCharArray()){
              if(c=='0'){
                  zero++;
              }else if(c== '1'){
                  one++;
              }
           }
           for(int i=m;i>=zero;i--){
               for(int j=n;j>=one;j--){
                   dp[i][j]=Math.max(dp[i][j],dp[i-zero][j-one]+1);
               }
           }
       }
       return dp[m][n];
    }
}

评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值