算法训练day42leetcode01背包问题 416. 分割等和子集

news/2024/7/7 19:34:59

01 背包

题目描述

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

题目分析

每一件物品其实只有两个状态,取或者不取,所以可以使用回溯法搜索出所有的情况,那么时间复杂度就是$o(2^n)$,这里的n表示物品数量。

所以暴力的解法是指数级别的时间复杂度。进而才需要动态规划的解法来进行优化!

  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]);

数据结构:

  • weight:一个向量,包含各个物品的重量。
  • value:一个向量,包含各个物品的价值。
  • bagweight:一个整数,代表背包的总容量。
  • dp:一个2D向量,其中dp[i][j]表示用前i个物品填充容量为j的背包时可以达到的最大价值。

算法步骤:

  1. 初始化:首先,使用物品0初始化dp数组的第一行。如果背包的容量大于等于物品0的重量,则背包可以装下物品0,所以对所有的j >= weight[0]dp[0][j]的值被设置为物品0的价值。
  2. 填充DP表:接下来,遍历每个物品(除了第一个已经处理过的物品),并对每个可能的背包容量进行考虑。对于每个物品i和每个容量j
    • 如果当前背包容量j小于物品i的重量,则无法包含当前物品,因此dp[i][j]的值就是不包含当前物品时的最大价值,即dp[i-1][j]
    • 如果当前背包容量可以容纳物品i,则需要决定是包含还是不包含当前物品以达到最大价值。这通过比较不包含当前物品(dp[i-1][j])和包含当前物品(dp[i-1][j-weight[i]] + value[i])时的价值来决定。选择这两种情况中的最大值作为dp[i][j]的值。

输出:

  • 通过查看dp数组的最后一个元素dp[weight.size() - 1][bagweight],可以得到使用给定物品填充指定容量背包的最大价值。

这种动态规划方法有效地解决了0/1背包问题,通过构建一个解决方案的表格,使得可以通过较小的子问题的解来构建出整个问题的解,从而避免了冗余的计算和指数级的复杂度。

acm模式代码

#include <iostream>
#include <vector>
using namespace std;

void test_2_wei_bag_problem1() {
    vector<int> weight = {1, 3, 4};
    vector<int> value = {15, 20, 30};
    int bagweight = 4;

    // 二维数组
    vector<vector<int>> dp(weight.size(), vector<int>(bagweight + 1, 0));


    // 初始化
    for (int j = 0 ; j < weight[0]; j++) {  // 当然这一步,如果把dp数组预先初始化为0了,这一步就可以省略,但很多同学应该没有想清楚这一点。
        dp[0][j] = 0;
    }
    for (int j = weight[0]; j <= bagweight; j++) {
        dp[0][j] = value[0];
    }

    // weight数组的大小 就是物品个数
    for(int i = 1; i < weight.size(); i++) { // 遍历物品
        for(int j = 0; j <= bagweight; j++) { // 遍历背包容量
            if (j < weight[i]) dp[i][j] = dp[i - 1][j];
            else dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - weight[i]] + value[i]);

        }
    }

    cout << dp[weight.size() - 1][bagweight] << endl;
}

int main() {
    test_2_wei_bag_problem1();
}

01背包一维

 题目分析

其实可以发现如果把dp[i - 1]那一层拷贝到dp[i]上,表达式完全可以是:dp[i][j] = max(dp[i][j], dp[i][j - weight[i]] + value[i]);

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

关于初始化,一定要和dp数组的定义吻合,否则到递推公式的时候就会越来越乱

这样才能让dp数组在递归公式的过程中取的最大的价值,而不是被初始值覆盖了

所以一维dp数组的背包在遍历顺序上和二维其实是有很大差异的!

acm模式代码

#include <iostream>
#include <vector>

void test_1_wei_bag_problem() {
    std::vector<int> weight = {1,3,4};
    std::vector<int> value = {15, 20 ,30};
    int bagweight = 4;

    //初始化
    std::vector<int> dp(bagweight + 1, 0);
    for (int i = 0; i < weight.size(); i++) {
        for (int j = bagweight;j >= weight[i]; j--) {
            dp[j] = std::max(dp[j], dp[j - weight[i]] + value[i]);

        }
    }
    std::cout << dp[bagweight] << std::endl;
}

int main() {
    test_1_wei_bag_problem();
}

416. 分割等和子集

题目描述

给你一个 只包含正整数 的 非空 数组 nums 。请你判断是否可以将这个数组分割成两个子集,使得两个子集的元素和相等。

示例 1:

输入:nums = [1,5,11,5]
输出:true
解释:数组可以分割成 [1, 5, 5] 和 [11] 。

示例 2:

输入:nums = [1,2,3,5]
输出:false
解释:数组不能分割成两个元素和相等的子集。

提示:

  • 1 <= nums.length <= 200
  • 1 <= nums[i] <= 100

acm模式代码


#include <iostream>
#include <vector>
using namespace std;

class Solution {
public:
    bool canPartition(vector<int>& nums) {
        int sum = 0;
        for (int num : nums) {
            sum += num;
        }
        if (sum % 2 != 0) return false; // Early return if the sum is odd.
        int target = sum / 2;
        vector<bool> dp(target + 1, false);
        dp[0] = true; // Base case: zero sum is always possible.

        for (int num : nums) {
            for (int j = target; j >= num; j--) {
                dp[j] = dp[j] || dp[j - num];
            }
        }
        return dp[target];
    }
};

int main() {
    Solution sol;
    vector<int> nums = {1, 5, 11, 5}; // Example input
    bool canPart = sol.canPartition(nums);
    if(canPart) {
        cout << "Can partition: YES" << endl;
    } else {
        cout << "Can partition: NO" << endl;
    }
    return 0;
}


http://lihuaxi.xjx100.cn/news/2088797.html

相关文章

手写简易操作系统(五)--获得物理内存容量

前情提要 上一章中我们进入了保护模式&#xff0c;并且跳转到了32位模式下执行。这一章较为简单&#xff0c;我们来获取物理内存的实际容量。 一、获得内存容量的方式 在Linux中有多种方法获取内存容量&#xff0c;如果一种方法失败&#xff0c;就会试用其他方法。其本质上是…

蓝桥杯刷题7

目录 1. 字母数 2. 列名 3. 大乘积 4. 最大连通 5. 星期几 1. 字母数 public class Main {public static void main(String[] args) {int num 2023;while(true) {String mInteger.toString(num,16);if(m.matches("^[a-f]$")){System.out.println(num);break;}n…

全景解析 Partisia Blockchain:以用户为中心的全新数字经济网络

在区块链世界中&#xff0c;以比特币、以太坊网络为代表的主流区块链奠定了该领域早期的基础&#xff0c;并让去中心化、点对点、公开透明以及不可逆成为了该领域固有的意识形态。事实上&#xff0c;过于透明正在成为区块链规模性采用的一大障碍&#xff0c;我们看到 90% 以上的…

14双体系Java学习之数组

数组 ★小贴士 数组中保存固定数量的值&#xff0c;声明数组时需要制定数组中元素的类型&#xff0c;数组的长度在创建数组时设定。 保存数据的数据结构有很多&#xff0c;Java的标准函数库中就包含了许多复杂的数据结构&#xff0c;比如map、tree和set&#xff0c;以后会讲解的…

【Echarts】曲线图上方显示数字以及自定义值,标题和副标题居中,鼠标上显示信息以及自定义信息

欢迎来到《小5讲堂》 大家好&#xff0c;我是全栈小5。 这是《前端》系列文章&#xff0c;每篇文章将以博主理解的角度展开讲解&#xff0c; 特别是针对知识点的概念进行叙说&#xff0c;大部分文章将会对这些概念进行实际例子验证&#xff0c;以此达到加深对知识点的理解和掌握…

基于PyTorch深度学习实战入门系列-Numpy基础全

Numpy的使用 导入Numpy模块 import numpy as np创建数组&#xff08;一维数组、小数数组、二维数组&#xff09; # 创建一个一维数组 n1 np.array([1, 2, 3]) # 创建一个含有小数的一维数组 n2 np.array([0.1, 0.2, 0.3]) # 创建一个简单的二维数组 n3 np.array([[1, 2], [3…

C语言分支和循环总结

文章目录 概要结构介绍不同结构的语句简单运用小结 概要 C语言中分为三种结构&#xff1a;顺序结构&#xff0c;选择结构&#xff0c;循环结构 结构介绍 顺序结构就是从上到下&#xff0c;从左到右等等&#xff1b;选择结构可以想象是Y字路口就是到了一个地方会有不同的道路…

LeetCode 1768. 交替合并字符串

文章目录 一、题目二、C 题解 一、题目 给你两个字符串 word1 和 word2。请你从 word1 开始&#xff0c;通过交替添加字母来合并字符串。如果一个字符串比另一个字符串长&#xff0c;就将多出来的字母追加到合并后字符串的末尾。 返回 合并后的字符串 。 示例 1&#xff1a; …