动态规划:0/1背包问题

01背包问题是一个经典的动态规划问题,它询问在给定的物品和背包容量下,如何选择物品使得背包中的物品总价值最大,同时保证不超过背包的容量限制。物品不能分割,每个物品只能选择放入或不放入背包。

问题定义

  • 输入:

    • 物品个数 ( n )
    • 背包容量 ( W )
    • 每个物品的重量 ( w[i] )
    • 每个物品的价值 ( v[i] )
  • 输出:

    • 背包所能装载的最大价值

动态规划解法思路

  1. 定义状态:

    • 定义 ( dp[i][j] ) 为从前 ( i ) 个物品中选择,总重量不超过 ( j ) 时的最大价值。
  2. 状态转移方程:

    • 如果不选择第 ( i ) 个物品,则 ( dp[i][j] = dp[i-1][j] )。
    • 如果选择第 ( i ) 个物品(前提是 ( j \geq w[i] )),则 ( dp[i][j] = dp[i-1][j-w[i]] + v[i] )。
    • 综合考虑上述两种情况,有:

      dp\[i\]\[j\] = \\max(dp\[i-1\]\[j\], dp\[i-1\]\[j-w\[i\]\] + v\[i\]) \\quad \\text{if } j \\geq w\[i

      ]
    • 如果 ( j < w[i] ),则只能选择不拿 ( i ) 物品的情况:( dp[i][j] = dp[i-1][j] )。
  3. 初始化条件:

    • ( dp[0][j] ) 为 0,因为没有物品时,最大价值为 0。
    • ( dp[i][0] ) 也为 0,因为容量为 0 时,不能装任何物品。
  4. 计算顺序:

    • 从 ( i = 1 ) 到 ( n ) 和 ( j = 1 ) 到 ( W )。
  5. 最终结果:

    • ( dp[n][W] ) 存储的是最大价值。

Java代码实现

下面是使用动态规划解决01背包问题的Java代码示例:

java 复制代码
public class Knapsack {
    public static int knapsack(int W, int n, int[] w, int[] v) {
        int[][] dp = new int[n+1][W+1];
        
        // 初始化dp数组
        for (int i = 0; i <= n; i++) {
            dp[i][0] = 0;
        }
        for (int j = 0; j <= W; j++) {
            dp[0][j] = 0;
        }
        
        // 动态规划填表
        for (int i = 1; i <= n; i++) {
            for (int j = 1; j <= W; j++) {
                if (j >= w[i-1]) {
                    dp[i][j] = Math.max(dp[i-1][j], dp[i-1][j-w[i-1]] + v[i-1]);
                } else {
                    dp[i][j] = dp[i-1][j];
                }
            }
        }
        
        // 返回最大价值
        return dp[n][W];
    }
    
    public static void main(String[] args) {
        int[] weights = {2, 3, 4, 5};
        int[] values = {3, 4, 5, 6};
        int capacity = 8;
        
        int maxProfit = knapsack(capacity, weights.length, weights, values);
        System.out.println("The maximum profit is " + maxProfit);
    }
}

在这段代码中,我们使用了一个二维数组 dp 来存储每一步的结果,并通过逐步计算填充这个表格,最终在 dp[n][W] 中得到背包能够装载

的最大价值。

相关推荐
凌波粒几秒前
LeetCode--二叉树层序遍历实战指南
算法·leetcode·职场和发展
洛水水4 分钟前
【力扣100题】48.乘积最大子数组
算法·leetcode·职场和发展
小小de风呀4 分钟前
de风——【从零开始学C++】(七):string类详解
开发语言·c++·算法
YL200404265 分钟前
042将有序数组转换为二叉搜索树
数据结构·算法·leetcode
qq_296553279 分钟前
矩阵对角线遍历:从暴力到最优的优雅解法
数据结构·线性代数·算法·青少年编程·矩阵·深度优先遍历
洛水水10 分钟前
【力扣100题】50.最长有效括号
算法·leetcode·职场和发展
数智工坊10 分钟前
【BLIP论文阅读】:统一视觉语言理解与生成的自举式预训练范式
论文阅读·人工智能·深度学习·算法·transformer
yyy(十一月限定版)11 分钟前
问题解决策略搜索训练3
算法
吃好睡好便好13 分钟前
在Matlab中绘制圆锥三维曲面图
开发语言·人工智能·学习·算法·matlab·信息可视化
兰令水15 分钟前
topcode【随机算法题】【2026.5.15打卡-java版本】
java·算法·leetcode