01背包问题和完全背包问题

转载 2016年08月31日 16:40:07

转载网址:http://blog.csdn.net/KangRoger/article/details/38864689

在hihocoder上面的题目中看到的这个问题,总结一下。先看01背包问题。

01背包问题:一个背包总容量为V,现在有N个物品,第i个 物品体积为weight[i],价值为value[i],现在往背包里面装东西,怎么装能使背包的内物品价值最大?

看到这个问题,可能会想到贪心算法,但是贪心其实是不对的。例如最少硬币找零问题,要用动态规划。动态规划思想就是解决子问题并记录子问题的解,这样就不用重复解决子问题了。

动态规划先找出子问题,我们可以这样考虑:在物品比较少,背包容量比较小时怎么解决?用一个数组f[i][j]表示,在只有i个物品,容量为j的情况下背包问题的最优解,那么当物品种类变大为i+1时,最优解是什么?第i+1个物品可以选择放进背包或者不放进背包(这也就是0和1),假设放进背包(前提是放得下),那么f[i+1][j]=f[i][j-weight[i+1]+value[i+1];如果不放进背包,那么f[i+1][j]=f[i][j]。

这就得出了状态转移方程:

f[i+1][j]=max(f[i][j],f[i][j-weight[i+1]+value[i+1])。

可以写出代码测试:

#include<iostream>
using namespace std;
#define  V 1500
unsigned int f[10][V];//全局变量,自动初始化为0
unsigned int weight[10];
unsigned int value[10];
#define  max(x,y)	(x)>(y)?(x):(y)
int main()
{
	
	int N,M;
	cin>>N;//物品个数
	cin>>M;//背包容量
	for (int i=1;i<=N; i++)
	{
		cin>>weight[i]>>value[i];
	}
	for (int i=1; i<=N; i++)
		for (int j=1; j<=M; j++)
		{
			if (weight[i]<=j)
			{
				f[i][j]=max(f[i-1][j],f[i-1][j-weight[i]]+value[i]);
			}
			else
				f[i][j]=f[i-1][j];
		}
	
	cout<<f[N][M]<<endl;//输出最优解

}

在hihocoder上面还讲到可以进一步优化内存使用。上面计算f[i][j]可以看出,在计算f[i][j]时只使用了f[i-1][0……j],没有使用其他子问题,因此在存储子问题的解时,只存储f[i-1]子问题的解即可。这样可以用两个一维数组解决,一个存储子问题,一个存储正在解决的子问题。

再进一步思考,计算f[i][j]时只使用了f[i-1][0……j],没有使用f[i-1][j+1]这样的话,我们先计算j的循环时,让j=M……1,只使用一个一维数组即可。

for i=1……N

for j=M……1

f[j]=max(f[j],f[j-weight[i]+value[i])

#include<iostream>
using namespace std;
#define  V 1500
unsigned int f[V];//全局变量,自动初始化为0
unsigned int weight[10];
unsigned int value[10];
#define  max(x,y)	(x)>(y)?(x):(y)
int main()
{
	
	int N,M;
	cin>>N;//物品个数
	cin>>M;//背包容量
	for (int i=1;i<=N; i++)
	{
		cin>>weight[i]>>value[i];
	}
	for (int i=1; i<=N; i++)
		for (int j=M; j>=1; j--)
		{
			if (weight[i]<=j)
			{
				f[j]=max(f[j],f[j-weight[i]]+value[i]);
			}			
		}
	
	cout<<f[M]<<endl;//输出最优解

}

在看完01背包问题,再来看完全背包问题:一个背包总容量为V,现在有N个物品,第i个 物品体积为weight[i],价值为value[i],每个物品都有无限多件,现在往背包里面装东西,怎么装能使背包的内物品价值最大?

对比一下,看到的区别是,完全背包问题中,物品有无限多件。往背包里面添加物品时,只要当前背包没装满,可以一直添加。那么状态转移方程为:

f[i+1][j]=max(f[i][j-k*weight[i+1]+k*value[i+1]),其中0<=k<=V/weight[i+1]

使用内存为一维数组,伪代码

for i=1……N

for j=1……M

f[j]=max(f[j],f[j-weight[i]+value[i])

和01背包问题唯一不同的是j是从1到M。01背包问题是在前一个子问题(i-1物品)的基础上来解决当前问题(i物品),向i-1种物品时的背包添加第i种物品;而完全背包问题是在解决当前问题(i种物品),向i种物品时的背包添加第i种物品。

代码如下:

#include<iostream>
using namespace std;
#define  V 1500
unsigned int f[V];//全局变量,自动初始化为0
unsigned int weight[10];
unsigned int value[10];
#define  max(x,y)	(x)>(y)?(x):(y)
int main()
{
	
	int N,M;
	cin>>N;//物品个数
	cin>>M;//背包容量
	for (int i=1;i<=N; i++)
	{
		cin>>weight[i]>>value[i];
	}
	for (int i=1; i<=N; i++)
		for (int j=1; j<=M; j++)
		{
			if (weight[i]<=j)
			{
				f[j]=max(f[j],f[j-weight[i]]+value[i]);
			}			
		}
	
	cout<<f[M]<<endl;//输出最优解

}



背包问题详解:01背包、完全背包、多重背包

参考链接: http://www.cnblogs.com/fengty90/p/3768845.html http://blog.csdn.net/mu399/article/details/7722...
  • na_beginning
  • na_beginning
  • 2017年03月17日 11:47
  • 2013

01背包问题和完全背包问题

在hihocoder上面两期的题目,一个01背包问题,一个完全背包问题。总结一下!...
  • KangRoger
  • KangRoger
  • 2014年08月27日 10:27
  • 50871

背包问题(01背包和完全背包)java求解

背包问题主要是指一个给定容量的背包、若干具有一定价值和重量的物品,如何选择物品放入背包使物品的价值最大。其中又分01背包和无限背包,这里主要讨论01背包,即每个物品最多放一个。而无限背包可以转化为01...
  • ls5718
  • ls5718
  • 2016年08月17日 09:37
  • 6748

完全背包问题的三种算法的java实现

  • 2016年10月21日 16:01
  • 7KB
  • 下载

背包问题详解(01背包,完全背包,多重背包,混合背包,二维费用背包……)

  • 2010年10月16日 22:21
  • 124KB
  • 下载

背包问题九讲(01背包,多重背包,完全背包等)

  • 2010年04月25日 14:08
  • 62KB
  • 下载

夕拾算法进阶篇:17)01背包和完全背包问题 (动态规划DP)

问题描述: 有n件物品,每件物品重量为w[i],价值为c[i],现有一个容量为V的背包,问如何选取物品放入背包,使得背包内的总价值最大? 解决思路: 令dp[i][v]表示前i件物品(0恰好装入容量为...
  • jiangpeng59
  • jiangpeng59
  • 2017年02月09日 23:01
  • 230

01背包及完全背包问题(51Nod - 1085,HDU 1114 Piggy-Bank)

51Nod - 1085题目: 在N件物品取出若干件放在容量为W的背包里,每件物品的体积为W1,W2……Wn(Wi为整数),与之相对应的价值为P1,P2……Pn(Pi为整数)。求背包能够容...
  • qq_38313974
  • qq_38313974
  • 2017年06月20日 14:35
  • 125

01背包+完全背包问题

背包问题已经有很多写得很好的博客,这里的背包问题只是一个个人学习总结,希望能够把问题说得更明白清楚易懂。贴下背包九讲问题的优秀博客网址:   http://blog.csdn.net/pi9nc/a...
  • tuyu233
  • tuyu233
  • 2016年09月24日 17:16
  • 74

hdu 1114 Piggy-Bank 动态规划 之完全背包问题

昨天听了羽哥的讲课,发现自己在dp问题上就是个小白~~ 切两道完全背包的题目来练习一下  继续学习背包九讲中 Piggy-Bank Time Limit: 2000/1000 MS (J...
  • yangshuolll
  • yangshuolll
  • 2012年12月15日 22:58
  • 1281
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:01背包问题和完全背包问题
举报原因:
原因补充:

(最多只允许输入30个字)