动态规划之01背包问题
创始人
2024-05-26 02:09:15
0

背包练习网址https://www.luogu.com.cn/contest/92872 想要做题的话可以到这里面来进行完成(邀请码:r36l)。注:要输入邀请码才可以进入。

满篇都是干货,有详细的注释和代码,请放心观看。

这就是传说中的 01 背包问题,这个问题看到之后主要有两种思路:

一、贪心做法(错误想法)

        这道题如果没有学过 01 背包问题的话,很容易想成一个贪心的问题,就是讲他的 “性价比"
 从高到低排序(这里的“性价比”指的是 \frac{v_i}{w_i} ),但是我们很容易发现这是错误的,因为将性价比较高的放在前面的话那么不可以尽量的吧空间占用完,所以我们可以显然的发现,这样的方法是错误的,但是如果题目的数据比较水的话还是可以骗很多分的。(在我建立的那个比赛里面,你们可以试一下用这种贪心策略去做,不出意外的话是拿不到满分的)。

        所以这种做法是错误的。

二、01背包问题做法(朴素版本)

        01 背包问题基本上是比较简单的 DP 问题。

        我们通过普通的做 DP 的思路,得先想想应该怎么定义 DP 的状态,我们应该怎么做呢?

        第一步:

        通过以往做 DP 的经验,那么可以不是很困难的得出定义 :dp[i][j] 为选取前  i 个物品,使得总重量不超过 j 的最大总价值(这个是很重要的点,如果在做动态规划的时候没有想出这个点,那么基本上这道题就没有什么希望了)。

        第二步:

        这个 DP 数组是否需要加初始化,但是是否定的,因为这道题的 dp 数组一直都在不断的更新,所以这里也不会溢出或者判断错误,可以得出结论:这里的 dp  数组不需要初始化!

        第三步:

        第三步是最难的一步,也是我主要要讲的一步,这一步就是支撑整个代码的“顶梁柱”,那就是:状态应该怎么转移呢

        其实可以发现:枚举每个物品,比如当前枚举到了第 i 个物品,可以发现一共有两种情况:
        一、不选第 i 个物品,那么 dp[i][j] = dp[i-1][j]

        二、选第 i 个物品,那么可以发现 dp[i][j] = dp[i-1][j-v[i]]+w[i]

        解释一下第二种的理解:在前 i-1 个物品中,是总重量不超过 j-v[i] (因为当前枚举到了第 i 个物品,总重量为 j,当前的物品重量为 v[i] ,所以可以发现当前的重量就不可以超过 j-v[i]),所以答案不就呼之欲出了。。(注意最后要加上当前这个物品的总价值)

        总结一下:这道题的状态转移方程式是:dp[i][j]=max(dp[i-1][j],dp[i-1][j-v[i]]+w[i])

        然后这里还要注意一个点,就是在 j-v[i] 的时候有可能会出现负下标,所以需要特判一下 j>=v[i],这就是一个推导状态转移方程式的一个简单的解析,所以就可以写出以下代码(代码中也有较为详细的解释,我在这里就不多说了):

/*f[i][j] = f[i-1][j-v[i]]+w[i]
价值等于前i-1个物品的总容量不超过(j-v[i])的总价值
j-v[i]的含义:最大的容量为j,然而要放进i这个物品,i物
品的重量为v[i],所以前i-1个物品的最大容量只能是(j-v[i])。*/#include 
#include using namespace std;
const int N = 1010;int n, m;//m就是题目中的v
int v[N], w[N], f[N][N];int main(){scanf("%d%d", &n, &m);for(int i = 1; i <= n; i ++ ) scanf("%d%d",&v[i],&w[i]);for(int i = 1; i <= n; i ++ )for(int j = 0; j <= m; j++){f[i][j] = f[i-1][j];if(j >= v[i]) f[i][j] = max(f[i][j], f[i-1][j-v[i]]+w[i]);}printf("%d\n", f[n][m]);return 0;
}

三、01背包问题做法(一维数组优化版本)

        这种方法会讲的稍微快一点,因为这种方法就是思路二的一个优化版本,所以建议可以先把思路二的想法先看懂,然后再看这里的解析。

        还是熟悉的 DP 三部曲:

        第一步:

         依旧是定义 DP 数组,这里定义成一维的数组,那么就可以轻易的定义出: dp[i] 表示不超过 i 的最大总价值。

        第二步:

         这个 DP 数组是否需要初始化,也就是说他会不会超出边界或者得到不正确的答案,可以简单的画图模拟一下,发现是明显不会覆盖或超出边界的,所以这里的 DP 数组依旧不需要初始化。

        第三步:

         这里我依旧要特别重要的讲一下,其实可以发现不论是二维数组还是一维数组,他们的状态转移其实都是差不多的,所以其实可以直接将上面推出来的那一个状态转移方程给移动到下面来。

        二维数组的状态转移:dp[i][j]=max(dp[i-1][j],dp[i-1][j-v[i]]+w[i]),那么一维数组就可以在二维数组的基础上进行“直接去掉一维”的操作,所以可以得出一维数组的状态转移方程式:dp[j]=max(dp[j],dp[j-v[i]]+w[i]),其实在这里不是很好说明它是对的,所以这里我给大家举一个例子:

比如这里输入三个物品和一个容量为 2 的背包,他们的 v[i] 分别是10,90,100w[i]分别是1,1,2.

然后可以自己尝试一下,发现其实这个答案是错误的。

        为什么是错误的呢?

        其实,并不是状态转移方程错了,而是循环的顺序错了,应该把循环反过来,才可以保证它的正确性,只要稍微想一下可以发现确实如此,因为从后往前进行查找,才不会出现错误的、或者重复的情况,从而就保证了现在这种策略的准确性。

        然后给出参考代码:

#include 
#include 
#include 
#include using namespace std;const int N = 1010;
int n, m, v[N], w[N], f[N];int main(){scanf("%d%d", &n, &m);for(int i = 1; i <= n; i++) scanf("%d%d", &v[i], &w[i]);for(int i = 1; i <= n; i++)for(int j = m; j >= v[i]; j--){/*注意:这里的循环变量要从0开始,因为
这里的这个j表示的是价值不超过j,所以j可以是0*/f[j] = max(f[j], f[j-v[i]]+w[i]);//我们可以发现每次f[j]数组都要依靠前一个,但是这里明显反了,所以要把循环倒过来}printf("%d\n", f[m]);return 0;
}

        注1:在上面给出的两则代码中我在文章中所写的 dp 函数我都用的是 f 函数代替,如有观看的效果不佳,请谅解。

        注2:如有疑问可以在洛谷上私信我也可以就在评论求进行提问,只要我还没有 AFO,都会尽力在 7 日内回复消息,请耐心等待。

谢谢大家支持,这是我的第一篇 CSDN 中的 DP 博客,如果有阅读的不便或者手滑错误请谅解,并且私信/在评论区指出,我会第一时间改进的!!!

觉得博主写的不错的关注支持一下吧!我会继续努力的~

      

相关内容

热门资讯

因存在安全隐患 福特公司召回约... 央视记者当地时间5月14日获悉,福特汽车公司宣布,由于存在可能导致车辆在行驶过程中丧失制动功能、增加...
“流动的博物馆”进校园传薪火 本报讯(记者 宁亚琴 摄影报道)当满载文物知识的流动展板进入校园,一场跨越时空的文化对话就此开启。5...
泽连斯基:基辅已准备好进行“任... 转自:财联社【泽连斯基:基辅已准备好进行“任何形式的谈判”以结束与俄罗斯的冲突】财联社5月15日电,...
公益金赋能伊春市青少年冰雪赛事 黑龙江省伊春市地处小兴安岭腹地,作为北方知名的森林生态旅游城市,冰雪旅游一直是“林都”伊春最亮眼的名...
“当时我还以为方向盘失控了” ... 平面示意图 “晚上经过这里变道,车身都要剧烈振动一下,好影响行车安全哦。”近日,有读者向重庆晨报反映...
“青春同行,公益筑梦”爱心传递... 与此同时,一场非凡的公益之旅也在同步进行。“青春同行,公益筑梦”活动,于5月6日至31日在28省市火...
阿尼玛卿黄河水源涵养地科学考察... 本报讯(记者 师晓琼)5月13日,由三江源生态保护基金会、三江源国家公园管理局、五矿国际信托有限公司...
抽签结果出炉 北京时间5月14日,2027年男篮世界杯预选赛抽签仪式在卡塔尔多哈举行,中国男篮最终被分在B组。和中...
欧洲三大股指收盘小幅下跌 德国... 转自:财联社【欧洲三大股指收盘小幅下跌 德国DAX指数跌0.47%】财联社5月15日电,欧洲三大股指...
36岁宋轶现身央视端午晚会录制... 5月14日晚,根据录制现场的路透信息,目前确定参与的艺人包括宋轶(白旗袍古典造型获赞)、刘宇(国风舞...
特朗普中东行 “绕不开”以色列 美国总统唐纳德·特朗普5月13日至16日访问沙特阿拉伯、卡塔尔和阿拉伯联合酋长国三国,此行将不会到访...
频繁的器材变革让国乒遭遇“多哈... 2025多哈世乒赛开幕进入倒计时,中国乒乓球队已经抵达多哈,提前进行适应性训练。澳门世界杯男单丢冠之...
这场家风主题活动上,近500岁... 转自:上观新闻奉贤区南桥镇吴塘村的“明代牡丹苑”内有一株距今近500年的古牡丹。据史料记载,明代书画...
评论丨如何把握美国通胀当前走势...   孙长忠(清华大学全球私募股权研究院研究员)  美国劳工部5月13日公布4月CPI通胀数据显示:4...
俄罗斯库尔斯克州发生交通事故 ... 当地时间5月14日,俄罗斯库尔斯克州州长发布消息称,该州一地发生了严重交通事故,客车与轿车相撞,已致...
美元预测:周四零售销售数据会将...   来源:汇通网  周三(5月14日)欧盘时段,美元指数(DXY)盘中跌至100.58,下跌0.40...
欧洲债市:意大利与德国国债利差...   意大利与德国国债利差一度跌破100个基点,为2021年9月以来首次出现,反映风险偏好改善。  意...
朝阳无限社区邻里节暖心启幕 转自:劳动午报 近日,朝阳区八里庄街道朝阳无限社区举办“爱满无限 睦邻同心 共筑美好”主题邻里节,通...
马来西亚18岁中国留学生绑架案... 5月13日,两名中国留学生在马来西亚伪造绑架,骗取父母500万元赎金,被警方逮捕。5月14日,曾协助...
健康证不“健康” 不是小事 转自:劳动午报 我们日常生活中接触到的厨师、外卖配送员、护工、月嫂等从业人员,每年都至少要进行一次相...