动态规划之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 博客,如果有阅读的不便或者手滑错误请谅解,并且私信/在评论区指出,我会第一时间改进的!!!

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

      

相关内容

热门资讯

百事创始人与可口可乐的爱情故事... 百事创始人与可口可乐的爱情故事?可乐与百事的爱情故事……世界上第一瓶可乐诞生于1886年,独特的口感...
龙珠Z悟空的遗产1怎么不能玩 龙珠Z悟空的遗产1怎么不能玩进入后,一直按A键就这样了,按回车和空白也没反应可以玩的,我本来也不能玩...
求此图出处 求此图出处应该是京阿尼动画《Free!》的同人图……人鱼是动画中的男主七濑遥人鱼的梗是来自动画第二季...
现在教育的话题很多,我们到底需... 现在教育的话题很多,我们到底需要怎样的教育?现代的教育,主要放在提升孩子的能力上
唐玄宗那么宠爱杨贵妃,为什么不... 唐玄宗那么宠爱杨贵妃,为什么不封她为皇后?唐玄宗虽然宠爱杨贵妃,但毕竟皇后没有错,不可能将皇后废掉,...
请问怎么乘坐地铁,方法,步骤? 请问怎么乘坐地铁,方法,步骤?首先我没见过,更没有坐过,所以我要问怎么坐地铁啊???其次广州不坐地铁...
两小时内告诉我答案!!! 我的... 两小时内告诉我答案!!! 我的喜欢 运动 看电影 读书 用韩语怎么翻译??저의 취미는 운동, 영화...
有没有主角实力强大却隐藏起来只... 有没有主角实力强大却隐藏起来只想平凡的过完生活的小说巜我真的长生不老》主角高中生,喜欢古典文化和同班...
郭德纲的长篇单口相声《今古奇观... 郭德纲的长篇单口相声《今古奇观》系列什么时候能完?听说是跟一个外地的出版社合作出版的,又没有消息灵通...
秋天的特点是什么? 秋天的特点是什么?秋季有三大特点:干燥、寒冷、果实成熟季。首先来说,干燥的天气,是秋天的必经过程,夏...
如何改写一个伤感的故事? 如何改写一个伤感的故事?一般故事的结局是最能体现伤感的。所以修改这个故事的结局。就能改掉这个伤感的故...
自然哲学是指什么 自然哲学是指什么自然哲学是指什么自然科学的前身 用哲学方法探求世界基本规律哲学的英文单词是philo...
考生报考教师招聘各年级段所需要... 考生报考教师招聘各年级段所需要的学历要求分别是什么?如果是报考幼师资格证需具有幼儿师范学校毕业或高等...
我表妹马上过20岁生日了,我想... 我表妹马上过20岁生日了,我想送首藏头诗或者藏尾诗给她,她的名字叫:沈明玉沈府佳人美无比,明月高照玉...
形容人讨厌成语 形容人讨厌成语 面目可憎miàn mù kě zēng[释义] 面目:面貌;可憎:令人厌恶。相貌...
为什么一听到声音就特别烦躁,有... 为什么一听到声音就特别烦躁,有时候还会控制不住想哭?是不是你最近压力太大了?去看看心理医生吧。要不然...
有没有知道名家描写四季的文章? 有没有知道名家描写四季的文章?只要是家描写四季的文章都行,大家帮找找吧~最好是夏天和冬天的~大家帮帮...
学习计划书的格式 学习计划书的格式急急急急急啊学习计划书格式分三块:1.学习计划书名称仔细回顾一下自己的学习情况,找出...
四年级下册暑假生活指导参考答案... 四年级下册暑假生活指导参考答案山东教育出版社(英语)是2011年的 是帮助tony闯关的。急急...
《翼年代记之东京默示录》主要讲... 《翼年代记之东京默示录》主要讲什么的讲的就是故事讲小樱及小狼来自玖楼国(在日文中,《魔卡少女樱》的大...