01背包问题

问题:有N件物品和一个容量为V的背包。第i件物品的费用是c[i],价值是w[i]。求解将哪些物品装入背包可使价值总和最大。

分析:

这是最基础的背包问题,特点是:每种物品仅有一件,可以选择放或不放。

用子问题定义状态:即f[i][v]表示前i件物品恰放入一个容量为v的背包可以获得的最大价值。则其状态转移方程便是:

f[i][v]=max{f[i-1][v],f[i-1][v-c[i]]+w[i]}

这个方程非常重要,基本上所有跟背包相关的问题的方程都是由它衍生出来的。所以有必要将它详细解释一下:“将前i件物品放入容量为v的背包中”这个子问题,若只考虑第i件物品的策略(放或不放),那么就可以转化为一个只牵扯前i-1件物品的问题。如果不放第i件物品,那么问题就转化为“前i-1件物 品放入容量为v的背包中”,价值为f[i-1][v];如果放第i件物品,那么问题就转化为“前i-1件物品放入剩下的容量为v-c[i]的背包中”,此时能获得的最大价值就是f[i-1][v-c[i]]再加上通过放入第i件物品获得的价值w[i]。

优化:

以上方法的时间和空间复杂度均为O(VN),其中时间复杂度应该已经不能再优化了,但空间复杂度却可以优化到O。

先考虑上面讲的基本思路如何实现,肯定是有一个主循环i=1..N,每次算出来二维数组f[i][0..V]的所有值。那么,如果只用一个数组 f[0..V],能不能保证第i次循环结束后f[v]中表示的就是我们定义的状态f[i][v]呢?f[i][v]是由f[i-1][v]和f[i-1] [v-c[i]]两个子问题递推而来,能否保证在推f[i][v]时(也即在第i次主循环中推f[v]时)能够得到f[i-1][v]和f[i-1] [v-c[i]]的值呢?事实上,这要求在每次主循环中我们以v=V..0的顺序推f[v],这样才能保证推f[v]时f[v-c[i]]保存的是状态 f[i-1][v-c[i]]的值。伪代码如下:

for i=1..N
    for v=V..0
        f[v]=max{f[v],f[v-c[i]]+w[i]};

代码实现:

 /***************01背包问题******************/
#include <iostream> using namespace std;
#define INF -65536
const int V=;//定义体积的最大值;
const int T=;//定义商品的数目;
int f[V+];
//#define EMPTY
int w[T]={,,,,};//商品的价值;
int c[T]={,,,,};//商品的体积;
int package()
{
#ifdef EMPTY//可以不装满
for(int i=;i<=V;i++)//条件编译,表示可以不存储满
{
f[i]=;
}
#else//必须装满
f[]=;
for(int i=;i<=V;i++)//条件编译,表示必须存储满
{
f[i]=INF;
}
#endif // EMPTY
for(int i=;i<T;i++)
{
for(int v=V;v>=c[i];v--)
{
f[v]=max(f[v],f[v-c[i]]+w[i]);
}
}
return f[V];
}
int main()
{
int temp;
temp=package();
cout<<temp<<endl;
return ;
}

我们看到的求最优解的背包问题题目中,事实上有两种不太相同的问法。有的题目要求“恰好装满背包”时的最优解,有的题目则并没有要求必须把背包装满。一种区别这两种问法的实现方法是在初始化的时候有所不同。

如果是第一种问法,要求恰好装满背包,那么在初始化时除了f[0]为0其它f[1..V]均设为-∞,这样就可以保证最终得到的f[N]是一种恰好装满背包的最优解。

如果并没有要求必须把背包装满,而是只希望价格尽量大,初始化时应该将f[0..V]全部设为0。

为什么呢?可以这样理解:初始化的f数组事实上就是在没有任何物品可以放入背包时的合法状态。如果要求背包恰好装满,那么此时只有容量为0的背包可能被价值为0的nothing“恰好装满”,其它容量的背包均没有合法的解,属于未定义的状态,它们的值就都应该是-∞了。如果背包并非必须被装满,那么 任何容量的背包都有一个合法解“什么都不装”,这个解的价值为0,所以初始时状态的值也就全部为0了。

明:

最新文章

  1. 搭建 Windows Server 2003 + IIS6.0 + FastCGI + PHP5.3.29 + MySQL5.5.38 + Memcached1.2.6
  2. 关键字nullable,nonnull,null_resettable,_Null_unspecified详解
  3. a different object with the same identifier value was already associated with the session:错误;
  4. Java——Selector
  5. centos6.5 mysql开机启动
  6. ASP代码审计一枚
  7. 妙味WEB前端开发全套视频教程+项目实战+移动端开发(99G)
  8. data-&quot;mit.edu-Thinking In C++&quot;
  9. Sqoop实现自定义job的增量导入
  10. 第一个UI脚本--python+selenium
  11. spring中得到servletContext对象方法
  12. 模板的Traits
  13. iOS 自动构建套件 - flow.ci + fir.im + Coding
  14. win10 uwp 九幽图床
  15. 这一次,VR离我们真的很近
  16. 测试highlightjs主题1
  17. photoshop实现倾斜图片的修正
  18. WEB应用打成jar包全记录
  19. Linux下ftp安装配置及三种用户的验证
  20. Node.js使用MySQL的连接池

热门文章

  1. UVa 10534 DP LIS Wavio Sequence
  2. HDU 3045 DP 斜率优化 Picnic Cows
  3. JSON Extractor/jp@gc - JSON Path Extractor 举例
  4. HDU 5483 Nux Walpurgis
  5. HDU 3271 SNIBB
  6. 长沙理工大学第十二届ACM大赛-重现赛
  7. CSU-1336: Interesting Calculator,最短路思想!
  8. 快速排序php
  9. *AtCoder Regular Contest 096F - Sweet Alchemy
  10. msp430项目编程50