代码随想录算法训练营第三十五天 | 01背包问题 二维,01背包问题 一维,416. 分割等和子集

三十五天打卡,背包问题入门,之前做过还是比较容易的


46.卡码网【携带研究材料】

题目链接

解题过程

  • 采用二维bp,bp[i][j]的含义是,背包容量为j,只装序号为0~i的物品时能装的最大价值
  • 若背包容量大于等于当前物品的重量,可以选择装或者不装
    • dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - weight[i]] + value[i]);
  • 若背包容量小于当前物品的重量,只能选择不装
    • dp[i][j] = dp[i - 1][j];

二维dp

#include<iostream>
#include<vector>
using namespace std;
int main() {
    int m, n;
    cin >> m >> n;
    vector<int>weight(m);
    vector<int>value(m);
    for (int i = 0; i < m; i++) {
        cin >> weight[i];
    }
    for (int i = 0; i < m; i++) {
        cin >> value[i];
    }
    vector<vector<int>>dp(m, vector<int>(n + 1));
    for (int j = weight[0]; j <= n; j++) {
        dp[0][j] = value[0];
    }
    for (int i = 1; i < m; i++) {
        for (int j = 0; j <= n; j++) {
            if (weight[i] <= j) {
                dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - weight[i]] + value[i]);
            } else {
                dp[i][j] = dp[i - 1][j];
            }
        }
    }
    cout << dp.back().back();
    return 0;
    
}

解题过程

  • 采用一维dp,bp[j]的含义是,背包容量为j,能装物品的最大价值
  • dp[j] = max(dp[j], dp[j - weight[i]] + value[i]);
  • 倒序遍历是为了让所有物品只被放入一次

一维dp

#include<iostream>
#include<vector>
using namespace std;
int main() {
    int m, n;
    cin >> m >> n;
    vector<int>weight(m);
    vector<int>value(m);
    for (int i = 0; i < m; i++) {
        cin >> weight[i];
    }
    for (int i = 0; i < m; i++) {
        cin >> value[i];
    }
    vector<int>dp(n + 1);
    for (int i = 0; i < m; i++) {
        for (int j = n; j >= 0; j--) {
            if (j >= weight[i]) {
                dp[j] = max(dp[j], dp[j - weight[i]] + value[i]);
            }
        }
    }
    cout << dp.back();
    return 0;
    
}

416.分割等和子集

题目链接

解题过程

  • 采用背包问题解法,背包容量为sum / 2,物品的重量和价值都为num[i]
  • 背包如果正好装满,说明找到了总和为 sum / 2 的子集。
  • 本题中我们要使用的是01背包,因为元素我们只能用一次。
  • dp[j]表示 背包总容量(所能装的总重量)是j,放进物品后,背的最大重量为dp[j]

01背包

class Solution {
public:
    bool canPartition(vector<int>& nums) {
        int target = 0;
        int sum = 0;
        for (int n : nums) sum += n;
        if (sum % 2 == 1) return false;
        target = sum / 2;
        vector<int>dp(target + 1);
        for (int i = 0; i < nums.size(); i++) {
            for (int j = target; j >= 0; j--) {
                if (j >= nums[i]) {
                    dp[j] = max(dp[j], dp[j - nums[i]] + nums[i]);
                }
            }
        }
        return dp.back() == target;
    }
};
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值