1、问题描述
已知:有一个容量为V的背包和N件物品,第i件物品的重量是weight[i],收益是cost[i]。
条件:每种物品都有无限件,能放多少就放多少。
问题:在不超过背包容量的情况下,最多能获得多少价值或收益
举例:物品个数N = 3,背包容量为V = 5,则背包可以装下的最大价值为40.
----------------------------------------------
2、基本思路(直接扩展01背包的方程)
由于本问题类似于01背包问题,在01背包问题中,物品要么取,要么不取,而在完全背包中,物品可以取0件、取1件、取2件...直到背包放不下位置。因此,可以直接在01背包的递推式中扩展得到。
- 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;
- }
复杂度分析:
程序需要求解N*V个状态,每一个状态需要的时间为O(v/Weight[i]),总的复杂度为O(NV*Σ(V/c[i]))。
代码优化:
完全背包问题有一个很简单有效的优化,是这样的:若两件物品i、j满足c[i]<=c[j]且w[i]>=w[j],则将物品j去掉,不用考虑。
即,如果一个物品A是占的地少且价值高,而物品B是占地多,但是价值不怎么高,那么肯定是优先考虑A物品的。
这里代码略。
----------------------------------------------
3、转换为01背包问题求解(直接利用01背包)
思路 1、完全背包的物品可以取无限件,根据背包的总容量V和第i件物品的总重量Weight[i],可知,背包中最多装入V/Weight[i](向下取整)件该物品。因此可以直接改变第i件物品的总个数,使之达到V/Weight[i](向下取整)件,之后直接利用01背包的思路进行操作即可。
举例:物品个数N = 3,背包容量为V = 5。
拆分之前的物品序列:
拆分之后的物品序列:
根据上述思想:在背包的最大容量(5)中,最多可以装入1件物品一,因此不用扩展物品一。最多可以装入2件物品二,因此可以扩展一件物品二。同理,可以扩展一件物品三。
时间复杂度的分析:O(NNew*V),其中V表示扩展前背包容量,NNew表示扩展后物品的个数,NNew =Σ(V/Weight[i](向下取整))
思路 2、对物品进行拆分时,拆成二进制的形式。
具体思路:把第i种物品拆成费用为weight[i]*2^k、价值为w[i]*2^k的若干件物品,其中k满足weight[i]*2^k<=V。
思路:这是二进制的思想,因为不管最优策略选几件第i种物品,总可以表示成若干个2^k件物品的和。
这样把每种物品拆成O(log V/weight[i])件物品,是一个很大的改进。
举例:物品个数N = 3,背包总容量为V = 5。
拆分之前的物品序列:
拆分之后的物品序列:
为了和前面的例子保持一致,这里才用之前的例子,但是这个例子没有更好的说明二进制的拆分方法拆分的物品个数会少写。
假设物品A的重量为2,收益为3,背包的总重量为20。
根据第一种拆分,可以拆成10个物品,每一个物品的重量为2,收益为3。
根据第二种拆分方法,可以拆成4个物品,分别是物品一(重量为1*2,收益为3),物品二(重量为2*2,收益为6),物品三(重量为4*2,收益为12),物品四(重量为8*2,收益为24)。
时间复杂度的分析:O(NNEW*V),其中V表示扩展前背包容量,NNew表示扩展后物品的个数,NNew = Σ(log V/weight[i](向下取整))
代码:
- #include <iostream>
- #include <vector>
- #include <assert.h>
- using namespace std;
- /*
- f[v]:表示第i件物品放入容量为v的背包后,获得的最大容量
- f[v] = max(f[v],f[v - weight[i]] + value[i]);
- 初始条件:f[0] = 0;
- */
- const int N = 3;
- const int V = 20;//5
- int weight[N + 1] = {0,3,2,2};
- int Value[N + 1] = {0,5,10,20};
- int NNew = 0;
- vector<int> weightVector;
- vector<int> Valuevector;
- int f[V + 1] = {0};
- /*拆分物品*/
- void SplitItem()
- {
- //从1开始
- weightVector.push_back(0);
- Valuevector.push_back(0);
- //开始拆分
- int nPower = 1;
- for (int i = 1;i <= N;i++)
- {
- nPower = 1;
- while (nPower * weight[i] <= V)
- {
- weightVector.push_back(nPower * weight[i]);
- Valuevector.push_back(nPower * Value[i]);
- nPower <<= 1;
- }
- }
- }
- int Completeknapsack()
- {
- //拆分物品
- SplitItem();
- //转化为01背包处理
- NNew = weightVector.size() - 1;//多加了一个0,要减去
- for (int i = 1;i <= NNew;i++)//物品个数变化
- {
- for (int v = V;v >= weightVector[i];v--)//背包容量仍是V
- {
- f[v] = max(f[v],f[v - weightVector[i]] + Valuevector[i]);
- }
- }
- return f[NNew];
- }
- int main()
- {
- cout<<Completeknapsack()<<endl;
- system("pause");
- return 1;
- }
4、O(VN)的算法
伪代码
- for (int i = 1;i <= N;i++)
- {
- for (int v = weight[i];v <= V;v++)
- {
- f[v] = max(f[v],f[v - weight[i]] + Value[i]);
- }
- }
分析:这和01背包的伪代码很相似,在01背包的代码中,v变化的区间是逆序循环的,即[V,Weight[i]]。而这里,v变化的区间是顺序循环的,即为[Weight[i],V]。
原因:
再次给出定义:
f[i][v]表示把前i件物品放入容量为v的背包时的最大代价。
f[i-1][v-c[i]]表示把前i - 1件物品放入容量为v的背包时的最大代价.
在01背包中,v变化的区间是逆序循环的原因:要保证由状态f[i-1][v-c[i]]递推状态f[i][v]时,f[i-1][v-c[i]]没有放入第i件物品。之后,在第i循环时,放入一件第i件物品。
01背包的方程:
- f[i][v] = max(f[i - 1][v],f[i - 1][v - weight[i]] + Value[i])
在完全背包中,v变化的区间是顺序循环的原因:完全背包的特点是每种物品可选无限件,在求解加选第i种物品带来的收益f[i][v]时,在状态f[i][v-c[i]]中已经尽可能多的放入物品i了,此时在f[i][v-c[i]]的基础上,我们可以再次放入一件物品i,此时也是在不超过背包容量的基础下,尽可能多的放入物品i。
完全背包的方程:
- f[i][v] = max(f[i - 1][v],f[i][v - weight[i]] + Value[i]);
举例:
物品个数N = 3,背包总容量为V = 5。
物品信息:
完全背包:
分析:
i = 2,表示正在处理第2件物品。在求解f[2][4]时,如果要计算把第2件物品放入背包后的代价时,我们需要知道f[2][2],此时f[2][2]中已经尽全力放入第2件物品了(已经放入一件)。此时此刻还可以在放入一件第2件物品,在背包容量为4时,最多可以放入两件第二件物品。
总结下,f[i][v]:表示在背包容量为v时,尽全力的放入第i件物品的代价。f[i][v - weight[i]]:表示在背包容量为v - weight[i]时,尽全力的放入第i件物品的代价。因此由f[i][v - weight[i]]转换为f[i][v]时,也是在f[i][v - weight[i]]的基础上有加入了一件物品i。
为了节省保存状态的空间,可以直接使用一维数组保存状态。
代码:迭代方程:f[i][v] = max(f[i - 1][v],f[i][v - weight[i]] + Value[i]);
- #include <iostream>
- #include <vector>
- #include <assert.h>
- using namespace std;
- const int N = 3;
- const int V = 5;//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 = weight[i];v <= V;v++)
- {
- f[i][v] = max(f[i - 1][v],f[i][v - weight[i]] + Value[i]);
- }
- }
- return f[N][V];
- }
- int main()
- {
- cout<<Completeknapsack()<<endl;
- system("pause");
- return 1;
- }
代码:迭代方程:f[v] = max(f[v],f[v - weight[i]] + Value[i]);
- #include <iostream>
- using namespace std;
- const int N = 3;
- const int V = 5;//5
- int weight[N + 1] = {0,3,2,2};
- int Value[N + 1] = {0,5,10,20};
- int f[V + 1] = {0};
- int Completeknapsack()
- {
- f[0] = 0;
- for (int i = 1;i <= N;i++)
- {
- for (int v = weight[i];v <= V;v++)
- {
- f[v] = max(f[v],f[v - weight[i]] + Value[i]);
- }
- }
- return f[V];
- }
- int main()
- {
- cout<<Completeknapsack()<<endl;
- system("pause");
- return 1;
- }
转自 http://blog.csdn.net/insistgogo/article/details/11081025
相关推荐
解决完全背包问题时,同样可以使用动态规划,但状态转移方程需要进行适当调整,考虑物品可以被无限次选取的情况。 3. **多重背包问题** 多重背包是介于01背包和完全背包之间的一种问题,每种物品有限定的库存数量。...
问题可以分为0-1背包、完全背包和多重背包三种类型,每种类型有不同的解题策略。 1. 0-1背包:每种物品只能选择0个或1个放入背包。 2. 完全背包:每种物品可以无限次放入背包,只要不超过背包容量。 3. 多重背包:...
6. **完全背包与多重背包**:在实际问题中,可能会遇到每种商品可以无限购买(完全背包)或有限购买(多重背包)的情况。这些情况需要修改状态转移方程,以适应不同限制。 7. **回溯法**:除了动态规划,回溯法也是...
这个问题的难点在于物品的选择是二元决策,即每件物品要么被完全放入背包,要么完全不放入,因此得名0-1背包问题。 遗传算法是一种模拟自然选择和遗传机制的全局优化方法,它通过模拟生物进化过程中的“适者生存”...
C++编写的背包算法程序 cpp 动态规划
用 PHP 实现的 01 背包算法,参考了网上的相关 C++ 算法,用来方便 PHP 程序员改造使用,我是用它来实现在指定宽度的栏中整齐的排列一堆标签云,效果非凡且神奇,初次使用时一瞬间的确有这样的感觉。
输入物品数量n,报的容量m,每个物品的体积,每个物品的价值 输入:最大价值
贪心算法是一种优化策略,它在每...在背包问题中,贪心算法对于某些特定类型的问题(如完全背包、非递减价值密度的物品)能得到最优解,但对于一般部分背包问题,可能需要结合动态规划等更强大的工具来确保全局最优。
此问题有多种变体,如完全背包问题(可以无限次选择同一种物品)、0-1背包问题(每种物品只能选一次)等。 MATLAB作为一款强大的数值计算和图形处理软件,提供了丰富的工具和函数支持算法的实现。在这个项目中,...
### 贪心算法在背包问题中的应用及C语言实现 #### 一、贪心算法简介 贪心算法是一种在每个步骤中都选择局部最优解的策略,希望通过一系列的局部最优选择来达到全局最优解的目的。它适用于某些特定的问题类型,在...
本项目中,遗传算法被应用于解决经典的背包问题,这是一种典型的组合优化问题。在背包问题中,我们通常有一组物品,每件物品都有一个重量和价值,目标是在不超过背包总容量的前提下,选取物品以最大化总价值。 首先...
混合背包问题是一个更复杂的背包问题变种,它结合了0-1背包和完全背包的特点。在0-1背包问题中,每种物品只能选择放入背包一次或不放入;而在完全背包问题中,每种物品可以无限次地放入背包。混合背包则允许每种物品...
它描述的是这样的场景:我们有一组物品,每种物品都有一个重量和价值,我们需要选择一部分物品放入一个容量有限的背包中,使得放入背包的物品总重量不超过背包的容量,同时使这些物品的总价值最大。0-1背包问题的...
这个是背包算法。。解决背包问题的java代码。。。。。。。。。。。。。。。。。。。。。。
通过对“MH背包密码算法”的原理学习和源代码实践,不仅可以掌握一种非对称加密方法,还能锻炼C/C++编程能力。然而,密码学发展迅速,安全性更高的算法不断出现,因此,持续学习和了解最新密码学技术至关重要。同时...
用贪心算法解决多重背包问题的C++解决方法
10. **拓展应用**:背包问题有多种变体,如完全背包(每个物品可以无限数量放入)、多重背包(每个物品有有限数量)等,可以根据实际需求进行扩展。 通过学习和理解这些知识点,你将能够实现一个功能完善的C语言...
【标题】"背包问题PSO算法代码"涉及的是在计算机科学和优化技术中的经典问题——背包问题,结合了粒子群优化(PSO)算法来寻找解决方案。粒子群优化是一种受到鸟类群体行为启发的全局优化算法,它通过模拟群体中个体...
背包算法在计算机科学中的应用 背包算法是计算机科学中的一种经典算法,用于解决组合优化问题。该算法的提出可以追溯到1978年,Merkel和Hellman提出了该问题。该算法的主要思想是,在给定总重量的限制下,如何选择...
完全背包问题是一类典型的动态规划问题,在计算机科学和算法设计领域中有着广泛的应用。它和0-1背包问题类似,都需要在限定的背包容量内,选择物品装入背包以获得价值的最大化,不同的是,完全背包问题中每个物品...