我是靠谱客的博主 微笑歌曲,这篇文章主要介绍多重背包,现在分享给大家,希望可以做个参考。

原文链接:http://blog.csdn.net/insistgogo/article/details/11176693

1、问题描述

已知:有一个容量为V的背包和N件物品,第i件物品最多有Num[i]件,每件物品的重量是weight[i],收益是cost[i]。

问题:在不超过背包容量的情况下,最多能获得多少价值或收益

举例:物品个数N = 3,背包容量为V = 8,则背包可以装下的最大价值为64.


----------------------------------------------

2、基本思路(直接扩展01背包的方程)

由于本问题和完全背包很类似,这里直接给出方程。

复制代码
1
2
3
4
5
6
7
8
状态转移方程: f[i][v]:表示前i件物品放入重量为v的背包获得的最大收益 f[i][v] = max(f[i][v],f[i - 1][V - k * Weight[i]] + k * Value[i]); 其中0 <= k <= min(Num[i],V/Weight[i]);//这里和完全背包不同。 边界条件 f[i][0] = 0; f[v][0] = 0;
代码:

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
#include <iostream> using namespace std; const int N = 3;//物品个数 const int V = 8;//背包容量 int Weight[N + 1] = {0,1,2,2}; int Value[N + 1] = {0,6,10,20}; int Num[N + 1] = {0,10,5,2}; int f[N + 1][V + 1] = {0}; /* f[i][v]:表示把前i件物品放入容量为v的背包中获得的最大收益。 f[i][v] = max(f[i - 1][v],f[i - 1][v - k * Weight[i]] + K * Value[i]);其中1 <= k <= min(Num[i],V/Weight[i]) //初始化 f[i][0] = 0; f[0][v] = 0; */ int MultiKnapsack() { int nCount = 0; //初始化 for (int i = 0;i <= N;i++) { f[i][0] = 0; } for (int v = 0;v <= V;v++) { f[0][v] = 0; } //递推 for (int i = 1;i <= N;i++) { for (int v = Weight[i];v <= V;v++) { f[i][v] = 0; nCount = min(Num[i],v/Weight[i]);//是当前背包容量v,而不是背包的总容量 for (int k = 0;k <= nCount;k++) { f[i][v] = max(f[i][v],f[i - 1][v - k * Weight[i]] + k * Value[i]); } } } return f[N][V]; } int main() { cout<<MultiKnapsack()<<endl; system("pause"); return 1; }

复杂度分析:

程序需要求解N*V个状态,每一个状态需要的时间为O(v/Weight[i]),总的复杂度为O(NV*Σ(V/Weight[i]))。

3、转换为01背包问题求解(直接利用01背包)

思路 1、直接对每一件物品进行拆分成min(Num[i],V/Weight[i])件,之后在拆分后的集合上进行01背包的求解。



最后

以上就是微笑歌曲最近收集整理的关于多重背包的全部内容,更多相关多重背包内容请搜索靠谱客的其他文章。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(89)

评论列表共有 0 条评论

立即
投稿
返回
顶部