1.问题描述
有N种物品和一个容量为V的背包,每种物品都有无限件可以使用。放入第i种物品消耗的空间是Ci,得到的价值是Wi。求解:将那些物品装入背包,可使这些物品耗费的空间总和不超过背包容量,且价值最大。
举例:背包容量V=5,有三个物品N=3,重量分别为3,2,2;价值分别为5,10,20。
2.问题思路
这个问题和01背包其实是十分相似的,只不过是每种物品可以取0件,1件,2件,,,V/Ci件,其中V/Ci向下取整,因此可以得到它的规划方程
f[i][v]:表示前i件物品放入容量为v的容量中时的最大收益
递推式:
f[i][v] = max(f[i - 1][v],f[i - K * weight[i]] + K * Value[i]); 其中 1 <= K * weight[i] <= v,(v指此时背包容量)
//初始条件
f[0][v] = 0;
f[i][0] = 0;
代码如下:
#include <iostream>
#include <assert.h>
using namespace std;
/*
f[i][v]:前i件物品放入背包容量为v的背包获得的最大收益
f[i][v] = max(f[i - 1][v],f[i - 1][v - k * Wi] + k * Vi,其中 1<=k<= v/Wi)
边界条件
f[0][v] = 0;
f[i][0] = 0;
*/
const int N = 3;
const int V = 5;
int weight[N + 1] = {0,3,2,2};
int Value[N + 1] = {0,5,10,20};
int f[N + 1][V + 1] = {0};
int Completeknapsack()
{
//边界条件
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 = 1;v <= V;v++)
{
f[i][v] = 0;
int nCount = v / weight[i];
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<<Completeknapsack()<<endl;
system("pause");
return 1;
}