介绍一下背包DP(Python)

背包问题概述

背包问题是一类经典的动态规划问题,分为0-1背包完全背包两种主要类型。核心思想是通过状态转移方程逐步填充一个二维数组(或优化后的一维数组),记录不同容量下的最优解。


0-1背包问题

问题描述 :给定物品的重量 weights 和价值 values,以及背包容量 capacity,每个物品只能选或不选,求最大价值。

动态规划解法

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

状态转移方程

复制代码
dp[i][j] = max(dp[i-1][j], dp[i-1][j-weights[i-1]] + values[i-1])  

(若 j >= weights[i-1],否则直接继承 dp[i-1][j]

Python代码

python 复制代码
def knapsack_01(weights, values, capacity):
    n = len(weights)
    dp = [[0] * (capacity + 1) for _ in range(n + 1)]
    
    for i in range(1, n + 1):
        for j in range(1, capacity + 1):
            if j >= weights[i-1]:
                dp[i][j] = max(dp[i-1][j], dp[i-1][j-weights[i-1]] + values[i-1])
            else:
                dp[i][j] = dp[i-1][j]
    return dp[n][capacity]
空间优化(一维数组)
python 复制代码
def knapsack_01_optimized(weights, values, capacity):
    dp = [0] * (capacity + 1)
    for i in range(len(weights)):
        for j in range(capacity, weights[i] - 1, -1):  # 逆向遍历
            dp[j] = max(dp[j], dp[j - weights[i]] + values[i])
    return dp[capacity]

完全背包问题

问题描述:与0-1背包类似,但每个物品可以选无限次。

状态转移方程

复制代码
dp[i][j] = max(dp[i-1][j], dp[i][j-weights[i-1]] + values[i-1])  

(注意与0-1背包的区别:第二项是 dp[i][...]

Python代码(空间优化版)

python 复制代码
def knapsack_unbounded(weights, values, capacity):
    dp = [0] * (capacity + 1)
    for i in range(len(weights)):
        for j in range(weights[i], capacity + 1):  # 正向遍历
            dp[j] = max(dp[j], dp[j - weights[i]] + values[i])
    return dp[capacity]

关键区别

  1. 遍历顺序

    • 0-1背包:内层循环逆向遍历(避免重复计算)。
    • 完全背包:内层循环正向遍历(允许重复选择)。
  2. 状态转移

    • 0-1背包依赖上一行 dp[i-1][...]
    • 完全背包依赖当前行 dp[i][...]

应用场景

  • 0-1背包:物品唯一性(如选/不选)。
  • 完全背包:物品无限供应(如硬币找零问题)。
相关推荐
2301_795099743 小时前
让 CSS Grid 自适应容器尺寸的动态布局方案
jvm·数据库·python
AI进化营-智能译站3 小时前
ROS2 C++开发系列12-用多态与虚函数构建可扩展的ROS2机器人行为模块
开发语言·c++·ai·机器人
呆萌的代Ma3 小时前
python读取并加载.env的配置文件
python
Muyuan19983 小时前
27.RAG 系统中的上下文充分性判断:从 Chunk 数量、FAISS 距离到 LLM Relevance Gate
python·django·pdf·fastapi·faiss
iCxhust3 小时前
微机原理实践教程(C语言篇)---A002流水灯
c语言·开发语言·单片机·嵌入式硬件·51单片机·课程设计·微机原理
莎士比亚的文学花园3 小时前
Linux驱动开发(3)——设备树
开发语言·javascript·ecmascript
图码3 小时前
如何用多种方法判断字符串是否为回文?
开发语言·数据结构·c++·算法·阿里云·线性回归·数字雕刻
U盘失踪了4 小时前
python curl转python脚本
开发语言·chrome·python
charlie1145141914 小时前
Linux 字符设备驱动:cdev、设备号与设备模型
linux·开发语言·驱动开发·c