蓝桥杯 Java B 组之背包问题(01背包、完全背包)

news/2025/2/24 13:25:12

Day 1:背包问题(01背包、完全背包)


📖 一、背包问题简介

背包问题是动态规划(DP)中一个经典的优化问题,涉及物品选择容量约束。通常分为以下几类:

  • 01 背包(0/1 Knapsack):每个物品只能选择 一次
  • 完全背包(Unbounded Knapsack):每个物品可以被选择无限次
  • 多重背包:每个物品有固定的数量,不能超过限制。
  • 分组背包:物品被分成多个组,每组只能选一个。

本次重点讨论01 背包完全背包,并通过 "分割等和子集""零钱兑换 II" 来加深理解。


📖 二、01 背包问题

问题描述: 给定 N 个物品和一个容量为 W 的背包,每个物品有一个 重量 w[i]价值 v[i]。求在不超过 W 的情况下,最大价值是多少?

🔹 01 背包的状态转移方程

定义 dp[i][j] 表示i 个物品在容量 j 下的最大价值

  1. 不选第 i 个物品dp[i][j] = dp[i-1][j]
  2. 选第 i 个物品(前提:j >= w[i]):dp[i][j] = dp[i-1][j - w[i]] + v[i]
  3. 最终答案dp[N][W]

🔹 代码实现(01 背包)

public class Knapsack01 {
    public int knapsack(int W, int[] weights, int[] values, int n) {
        int[][] dp = new int[n + 1][W + 1];

        for (int i = 1; i <= n; i++) {
            for (int j = 1; j <= W; j++) {
                dp[i][j] = dp[i - 1][j]; // 不选第 i 个物品
                if (j >= weights[i - 1]) {
                    dp[i][j] = Math.max(dp[i][j], dp[i - 1][j - weights[i - 1]] + values[i - 1]);
                }
            }
        }
        return dp[n][W];
    }

    public static void main(String[] args) {
        Knapsack01 knapsack = new Knapsack01();
        int[] weights = {2, 3, 4, 5};
        int[] values = {3, 4, 5, 6};
        int W = 5;
        int n = weights.length;
        System.out.println("最大价值: " + knapsack.knapsack(W, weights, values, n)); // 输出 7
    }
}

🔹 时间复杂度

O(n * W),其中 n 是物品数量,W 是背包容量。


📖 三、完全背包问题

问题描述: 与 01 背包 不同,完全背包允许每个物品选取无限次

🔹 完全背包的状态转移方程

  1. 不选第 i 个物品dp[i][j] = dp[i-1][j]
  2. k 次第 i 个物品(前提:j >= k * w[i]): dp[i][j]=max⁡(dp[i][j],dp[i][j−k×w[i]]+k×v[i])dp[i][j] = \max(dp[i][j], dp[i][j - k \times w[i]] + k \times v[i])

优化版

dp[i][j]=max⁡(dp[i−1][j],dp[i][j−w[i]]+v[i])dp[i][j] = \max(dp[i-1][j], dp[i][j-w[i]] + v[i])

注意:完全背包的状态转移是从 dp[i][j-w[i]] 来的,而不是 dp[i-1][j-w[i]],表示可以重复选取当前物品

🔹 代码实现(完全背包)

public class KnapsackComplete {
    public int knapsack(int W, int[] weights, int[] values, int n) {
        int[] dp = new int[W + 1];

        for (int i = 0; i < n; i++) {
            for (int j = weights[i]; j <= W; j++) { // 正序遍历
                dp[j] = Math.max(dp[j], dp[j - weights[i]] + values[i]);
            }
        }
        return dp[W];
    }

    public static void main(String[] args) {
        KnapsackComplete knapsack = new KnapsackComplete();
        int[] weights = {2, 3, 4, 5};
        int[] values = {3, 4, 5, 6};
        int W = 5;
        int n = weights.length;
        System.out.println("最大价值: " + knapsack.knapsack(W, weights, values, n)); // 输出 8
    }
}

🔹 时间复杂度

O(n * W),但使用了一维 dp 数组,空间复杂度从 O(n*W) 降到了 O(W)


📖 四、练习 1:分割等和子集(Subset Sum)

题目描述: 给定一个非负整数数组 nums,判断是否可以将其分割为两个子集,使得两个子集的和相等。

🔹 思路

  • 转化为 01 背包问题
    • 目标是找到一个子集,使得其和为 sum/2
    • 如果 sum 为奇数,则直接返回 false
    • 状态定义dp[j] 表示能否填满容量 j 的背包
    • 状态转移方程dp[j] = dp[j] || dp[j - nums[i]]

🔹 代码实现

import java.util.*;

public class PartitionEqualSubsetSum {
    public boolean canPartition(int[] nums) {
        int sum = Arrays.stream(nums).sum();
        if (sum % 2 != 0) return false; // 奇数直接返回 false

        int target = sum / 2;
        boolean[] dp = new boolean[target + 1];
        dp[0] = true;

        for (int num : nums) {
            for (int j = target; j >= num; j--) { // 01 背包,倒序遍历
                dp[j] = dp[j] || dp[j - num];
            }
        }
        return dp[target];
    }

    public static void main(String[] args) {
        PartitionEqualSubsetSum solution = new PartitionEqualSubsetSum();
        int[] nums = {1, 5, 11, 5};
        System.out.println(solution.canPartition(nums)); // 输出 true
    }
}

时间复杂度:O(n * sum/2)sum 是数组和。


📖 五、练习 2:零钱兑换 II(Coin Change II)

题目描述: 给定不同面额的硬币 coins 和一个总金额 amount,求总共有多少种方式可以凑成 amount

🔹 代码实现

public class CoinChange2 {
    public int change(int amount, int[] coins) {
        int[] dp = new int[amount + 1];
        dp[0] = 1; // 组合数初始化

        for (int coin : coins) {
            for (int j = coin; j <= amount; j++) { // 完全背包,正序遍历
                dp[j] += dp[j - coin];
            }
        }
        return dp[amount];
    }

    public static void main(String[] args) {
        CoinChange2 solution = new CoinChange2();
        int[] coins = {1, 2, 5};
        int amount = 5;
        System.out.println(solution.change(amount, coins)); // 输出 4
    }
}

时间复杂度:O(n * amount)


📖 六、总结

01 背包 vs 完全背包

背包类型状态转移
01 背包dp[i][j] = max(dp[i-1][j], dp[i-1][j-w[i]] + v[i])
完全背包dp[i][j] = max(dp[i-1][j], dp[i][j-w[i]] + v[i])

🎯 练习建议

  • 先熟练掌握 01 背包,再理解 完全背包 的正序遍历优化。
  • 多练习 变种题型,如 分割等和子集、零钱兑换 II

http://www.niftyadmin.cn/n/5864381.html

相关文章

49 set与map的模拟实现

目录 一、源码及框架分析 二、模拟实现map和set &#xff08;一&#xff09;复用红黑树的框架&#xff0c;并支持insert &#xff08;二&#xff09;支持迭代器的实现 &#xff08;三&#xff09;map支持 [ ] &#xff08;四&#xff09;整体代码实现 一、源码及框架分析…

【Go】Go wire 依赖注入

1. wire 简介 wire 是一个 Golang 的依赖注入框架&#xff08;类比 Spring 框架提供的依赖注入功能&#xff09; ⭐ 官方文档&#xff1a;https://github.com/google/wire 这里关乎到编程世界当中一条好用的设计原则&#xff1a;A用到了B&#xff0c;那么B一定是通过依赖注入的…

【CVPR2024-工业异常检测】PromptAD:与只有正常样本的少样本异常检测的学习提示

代码链接 摘要 摘要写作总结&#xff1a; 1.提出 两个关键点 &#xff08;视觉语言模型【模型】 少量工业异常检测【方向】&#xff09; 2.想要解决的问题 3.针对上述问题&#xff0c;本文提出了一种什么【方法】的什么【应用方面】方法【模型名】 4.具体讲方法的步骤 5.实验…

0基础玩转python(打怪升级篇)第一章1.1安装python编辑器

第一章 新手村 1.1合适的武器 &#xff08;安装python编辑器&#xff09; 新手村位于代码大陆的东北部&#xff0c;四面环山&#xff0c;高大的城墙外坐落着一条护城河。一位少年缓步走进城内&#xff0c;看到了城内集市热闹非凡。 你叫做“阿印” 是一位勇士 当前等级o Lv 最…

蓝桥杯学习笔记04-滑动窗口不定长(最短/最小)

题目来源 分享丨【题单】滑动窗口与双指针&#xff08;定长/不定长/单序列/双序列/三指针/分组循环&#xff09; - 力扣&#xff08;LeetCode&#xff09; 209. 长度最小的子数组 - 力扣&#xff08;LeetCode&#xff09; 题目要求大于等于 class Solution { public:int min…

AWS EC2加速型计算实例全解析:从vt1到p5,如何为AI算力选择最佳引擎?

在人工智能技术高速发展的今天&#xff0c;算力已成为驱动创新的核心动力。AWS EC2加速型计算实例家族凭借其强大的异构计算能力&#xff0c;正在重塑AI开发者的生产力边界。本文将深入解析从vt1.3xlarge到p5.48xlarge的全系列实例&#xff0c;带您找到最适合AI训练与推理的云端…

Python 高级特性-迭代器

目录 迭代器 小结 迭代器 我们已经知道&#xff0c;可以直接作用于for循环的数据类型有以下几种&#xff1a; 一类是集合数据类型&#xff0c;如list、tuple、dict、set、str等&#xff1b; 一类是generator&#xff0c;包括生成器和带yield的generator function。 这些可…

在 Mac ARM 架构上使用官方安装包安装 MySQL

在 Mac ARM 架构 (Apple Silicon&#xff0c;例如 M1, M2, M3 芯片) 上使用官方安装包安装 MySQL&#xff0c;步骤与在 Intel Mac 上类似&#xff0c;但需要确保下载的是 ARM 架构兼容的版本。以下是详细的安装步骤&#xff1a; 步骤 1: 下载 MySQL Community Server DMG 安装…