当前位置:网站首页 > 技术博客 > 正文

0-1背包问题与背包问题



背包问题泛指以下这一种问题:

给定一组有固定价值和固定重量的物品,以及一个已知最大承重量的背包,求在不超过背包最大承重量的前提下,能放进背包里面的物品的最大总价值。

这一类问题是典型的使用动态规划解决的问题,我们可以把背包问题分成3种不同的子问题:0-1背包问题、完全背包和多重背包问题。下面对这三种问题分别进行讨论。

1.0-1背包问题

0-1背包问题是指每一种物品都只有一件,可以选择放或者不放。现在假设有n件物品,背包承重为m。

对于这种问题,我们可以采用一个二维数组去解决:f[i][j],其中i代表加入背包的是前i件物品,j表示背包的承重,f[i][j]表示当前状态下能放进背包里面的物品的最大总价值。那么,f[n][m]就是我们的最终结果了。

采用动态规划,必须要知道初始状态和状态转移方程。初始状态很容易就能知道,那么状态转移方程如何求呢?对于一件物品,我们有放进或者不放进背包两种选择:

(1)假如我们放进背包,f[i][j] = f[i - 1][j - weight[i]] + value[i],这里的f[i - 1][j - weight[i]] + value[i]应该这么理解:在没放这件物品之前的状态值加上要放进去这件物品的价值。而对于f[i - 1][j - weight[i]]这部分,i - 1很容易理解,关键是 j - weight[i]这里,我们要明白:要把这件物品放进背包,就得在背包里面预留这一部分空间。

(2)假如我们不放进背包,f[i][j] = f[i - 1][j],这个很容易理解。

因此,我们的状态转移方程就是:f[i][j] = max(f[i][j] = f[i - 1][j] , f[i - 1][j - weight[i]] + value[i])  

当然,还有一种特殊的情况,就是背包放不下当前这一件物品,这种情况下f[i][j] = f[i - 1][j]。

下面是实现的代码:

特别的是,0-1背包问题还有一种更加节省空间的方法,那就是采用一维数组去解决,下面是代码:

我看过很多博客的描述,讲得都不太清楚:为什么要把第二层循环颠倒过来呢?我认为要理解这种方法,用图是最合适不过了,我在另外一个博客(http://blog.csdn.net/mu399/article/details/)找到了这样一个图:

这个表格对于理解0-1背包问题很有用,我们利用它来理解一下为什么要把第二层循环颠倒这个问题。考虑d9这一项,要求出这个状态,我们有可能利用到的就是e1到e8这8个状态,当我们把第二层循环颠倒过来时,当我们要求f[j]时,f[j -1]到f[1]还保存着下面一行的状态,因此可以采用一维数组解决。我们可以考虑一下假如不把第二层循环颠倒,当要求f[j]时,f[j - 1]到f[1]已经是同一行的状态了,根本没法求。

我在LeetCode上也曾做过类似的题目——120 Triangle(https://leetcode.com/problems/triangle/description/),也是把二维数组简化为一维数组去解决问题。

更新:

这是0-1背包最重要的部分,可以把它改为下面的更简洁的版本:

2.完全背包问题

完全背包问题是指每种物品都有无限件,具体的解法我也解释不清楚,只能先放代码,谈谈我的理解了:

i=1时,计算只放第一件物品的最大价值。

i=2时,计算加上第二件物品的最大价值(在只放第一件物品的前提下)

以此类推……

值得注意的是,第二层循环要从j=weight[i]开始,这个稍微理解一下即可。

我在别的博客看到了一段分析0-1背包问题和完全背包问题区别的话,觉得对理解这两个问题很有帮助,因此截图如下:

3.多重背包问题

多重背包问题限定了一种物品的个数,解决多重背包问题,只需要把它转化为0-1背包问题即可。比如,有2件价值为5,重量为2的同一物品,我们就可以分为物品a和物品b,a和b的价值都为5,重量都为2,但我们把它们视作不同的物品。

代码如下:

版权声明


相关文章:

  • 137端口是什么协议2025-01-08 23:30:05
  • 循环队列和顺序队列区别2025-01-08 23:30:05
  • java 集合类框架的最佳实践有哪些?2025-01-08 23:30:05
  • css中设置溢出隐藏2025-01-08 23:30:05
  • 计算机c语言基础知识2025-01-08 23:30:05
  • ubuntu自带输入法怎么用2025-01-08 23:30:05
  • 抢票 python2025-01-08 23:30:05
  • c解析json字符串2025-01-08 23:30:05
  • 彻底关闭137端口2025-01-08 23:30:05
  • 匿名内部类的用法2025-01-08 23:30:05