01背包问题:图表法带你快速理解动态规划解决01背包问题 附C++源码

0-1背包问题

所谓0-1背包问题,也就是给你一个重量为M的背包和n种物品,每种物品有一定的重量和价值,在每种物品均可装入背包1次或不装入(不能仅装入物品的一部分)且不超过背包载重量的前提下,问你怎样选择物品,使得装入背包的物品的总价值最大?

网上关于0-1背包问题的解决办法非常多,但是上来就给公式,我觉得对于初学者来说非常不好理解,目前我觉得最好的方式就是图表法来快速理解这个问题,当然大家如果有更好的方法欢迎在评论区分享。

分析

我们先从一个例子入手:
假如现在有一个背包能够承重5kg,有四个物品重量和价值如下:

物品重量/kg价值
物品①310
物品②240
物品③430
物品④150

思路:对于每个物品,我们计算背包当前状态是否可以存放,如果能存放,则比较放入该物品的总价值和未放之前的总价值,如果该物品放入后比之前的总价值更大,则放入该物品,反之则不放。这样,当前状态所得到的总价值就是最大的。后面每放一个物品,我们都在前面计算的总价值最大的基础上再进行上述重复操作。

如图,横轴为背包重量,我们让背包重量从0开始,慢慢增长到5,分析每一个重量下可放入物品的最大价值;纵轴为我们的物品编号,我们总共有四个物品。当背包重量为0时,没有物品能够放入,背包的价值也就为0,所以表格第一列是0;当物品为0时,即没有东西放入背包,此时背包的价值也都是0,所以表格第一行都是0.(把0考虑进来的目的是方便计算和分析)
在这里插入图片描述
然后我们先判断物品①放入背包中的情况,由上述可知物品①重量为3,价值为10,当背包重承重为1或者2时物品①无法放入,所以最大价值是0,当背包承重为3、4、5时,物品①可以放入,所以背包最大价值是10,表格数据情况如下:
在这里插入图片描述
接着我们再判断放入第二个物品的情况,物品②重量为2,价值为40:

  • 当背包承重为1时,物品②无法放入,背包总价值为0
  • 当背包承重为2时,刚好可以放入物品②,此时背包最大价值为40
  • 当背包承重为3时,不放物品②的最大价值由上述计算得出是10(即放入物品①),放入物品②的最大价值是40,我们选择放入物品②
  • 背包承重为4时跟承重为3的情况一模一样,背包的最大价值还是40
  • 当背包承重为5时,这个时候物品①和物品②可以同时放入,所以背包的最大价值为10+40=50
    在这里插入图片描述
    第三个物品重量为4,价值为30,所以
  • 当背包承重为1,2,3时,物品③都无法放入,根据上述计算可知,背包承重为1时,没有物品能放入,最大价值是0,当背包承重为2或者3时,背包的最大价值为40(即上述放入物品②)
  • 当背包承重为4时,有两种情况,一种是放入物品③,此时背包价值为30,第二种情况就是不放入物品③,不放入物品③且当背包重量为4的时候我们计算出背包的最大价值为40,所以我们选择不放入。
  • 当背包重量为5时,同理,两种情况,放入物品③价值为30,不放物品③且背包重量为5的最大价值是50(上述计算得出),所以背包的最大价值还是50.
    在这里插入图片描述
    第四个物品重量为1,价值为50,所以
  • 当背包承重为1时,我们放入物品④价值最大,最大价值为50
  • 当背包承重为2时,我们有两个选择,,一是放入物品④最大价值为50,二是不放入物品④,此时最大价值是40(由上述计算可得是放入物品②)
  • 当背包承重为3时,不放入物品④的最大价值是40,放入物品④的最大价值是90(因为此时物品④和物品②可以同时放入)
  • 当背包承重为4、5时,最大价值还是90
    在这里插入图片描述
    到这里,我们就通过画图表进行分析的方式得到了我们背包可放置物品的最大总价值!

由上面的分析过程和表格,不知道各位小伙伴们有没有看出上面规律,其实上面的表格我们完全可以把它看成一个二维数组,这个二维数组里面记录了背包承重的不同情况下放入不同物品所得的最大价值。

假如,我们把这个二维数组定义为dp[ i ][ j ],当背包重为0时,没有物品能够放得下,所以背包价值为0即dp[ i ][ 0]={0},当没有物品放进背包时,背包的价值也是0,也即dp[ 0 ][ j ]={0},当我们有物品需要放入时,根据上述我们可以得到如下公式来推导出最大价值:
在这里插入图片描述
在这里插入图片描述

源代码

关注公众号百宝菌并回复:01背包问题
即可免费获得C++源码!
或者从csdn获取付费资源,编程不易,敬请见谅!
资源链接:https://download.csdn.net/download/David_house/86791128

  • 10
    点赞
  • 42
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
好的,下面是用C语言实现动态规划解决01背包问题的代码: ```c #include <stdio.h> #include <stdlib.h> // 定义物品结构体 typedef struct Item { int weight; // 物品重量 int value; // 物品价值 } Item; // 动态规划函数 int knapsack(Item* items, int n, int W) { int i, w; int** K = (int**) malloc((n + 1) * sizeof(int*)); // 申请二维数组 for (i = 0; i <= n; i++) { K[i] = (int*) malloc((W + 1) * sizeof(int)); } // 填充二维数组 for (i = 0; i <= n; i++) { for (w = 0; w <= W; w++) { if (i == 0 || w == 0) { // 初始化第一行和第一列为0 K[i][w] = 0; } else if (items[i - 1].weight <= w) { // 当前物品可以装入背包 int val1 = items[i - 1].value + K[i - 1][w - items[i - 1].weight]; // 装入该物品的最大价值 int val2 = K[i - 1][w]; // 不装入该物品的最大价值 K[i][w] = (val1 > val2) ? val1 : val2; // 取最大值 } else { // 当前物品无法装入背包 K[i][w] = K[i - 1][w]; } } } int result = K[n][W]; // 获取最终结果 // 释放动态申请的内存 for (i = 0; i <= n; i++) { free(K[i]); } free(K); return result; } int main() { int W = 50; // 背包最大承重 Item items[] = {{10, 60}, {20, 100}, {30, 120}}; // 物品数组 int n = sizeof(items) / sizeof(items[0]); // 物品数量 int max_value = knapsack(items, n, W); // 动态规划求解 printf("The maximum value is %d\n", max_value); return 0; } ``` 上述代码中,我们定义了一个`Item`结构体,它包含了物品的重量和价值,然后我们通过`knapsack`函数进行动态规划求解,最后输出最大的价值。代码中使用了二维数组来存储物品的最大价值,其中`K[i][w]`示前i个物品在承重为w的情况下的最大价值。在求解过程中,我们首先初始化第一行和第一列为0,然后根据当前物品是否可以装入背包进行状态转移,最后得到最终结果。 需要注意的是,动态申请的内存需要在程序结束后手动释放,以避免内存泄漏。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

疯狂的小强呀

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值