背包问题(0-1背包,完全背包,多重背包知识概念详解)

 

通过:http://www.cnblogs.com/tanky_woo/archive/2010/07/31/1789621.html  

   http://blog.csdn.net/lyhvoyage/article/details/8545852改编而来

 

3种背包的简单概念:

0-1背包  (ZeroOnePack): 有N件物品和一个容量为V的背包。每种物品均只有一件

             第i件物品的费用是c[i],价值是w[i]。求解将哪些物品装入背包可使价值总和最大。

完全背包(CompletePack): 有N种物品和一个容量为V的背包,每种物品都有无限件可用

              第i种物品的费用是c[i],价值是w[i]。求解将哪些物品装入背包可使这些物品的费用总和不超过背包容量,且价值总和最大。

多重背包   (MultiplePack): 有N种物品和一个容量为V的背包。第i种物品最多有n[i]件可用

                每件费用是c[i],价值是w[i]。求解将哪些物品装入背包可使这些物品的费用总和不超过背包容量,且价值总和最大。

比较三个题概念,会发现不同点在于每种背包的数量,01背包是每种只有一件,完全背包是每种无限件,而多重背包是每种有限件。

——————————————————————————————————————————————————————————–

一、0-1背包:

有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件不放进去,这时所得价值为:f[i-1][v]

第二种是第i件放进去,这时所得价值为:f[i-1][v-c[i]]+w[i]

(第二种是什么意思?就是如果第i件放进去,那么在容量v-c[i]里就要放进前i-1件物品)

最后比较第一种与第二种所得价值的大小,哪种相对大,f[i][v]的值就是哪种。

(这是基础,要理解!)

 

V=10,N=3,c[]={3,4,5}, w={4,5,6}

(1)背包不一定装满

      计算顺序是:从右往左,自上而下:因为每个物品只能放一次,前面的体积小的会影响体积大的

      

(2)背包刚好装满    

      计算顺序是:从右往左,自上而下。注意初始值,其中-inf表示负无穷

      

这里是用二位数组存储的,可以把空间优化,用一位数组存储。

用f[0..v]表示,f[v]表示把前i件物品放入容量为v的背包里得到的价值。把i从1~n(n件)循环后,最后f[v]表示所求最大值。

*这里f[v]就相当于二位数组的f[i][v]。那么,如何得到f[i-1][v]和f[i-1][v-c[i]]+w[i]?(重点!思考)
首先要知道,我们是通过i从1到n的循环来依次表示前i件物品存入的状态。即:for 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]};
测试数据:
10,3
3,4
4,5
5,6

         

这个图表画得很好,借此来分析:

C[v]从物品i=1开始,循环到物品3,期间,每次逆序得到容量v在前i件物品时可以得到的最大值。(请在草稿纸上自己画一画

 

这里以一道题目来具体看看:

 

题目:http://acm.hdu.edu.cn/showproblem.php?pid=2602

 

代码在这里:http://www.wutianqi.com/?p=533

 

 

二、完全背包:

有N种物品和一个容量为V的背包,每种物品都有无限件可用。第i种物品的费用是c[i],价值是w[i]。求解将哪些物品装入背包可使这些物品的费用总和不超过背包容量,

且价值总和最大。

完全背包按其思路仍然可以用一个二维数组来写出:

f[i][v]=max{f[i-1][v-k*c[i]]+k*w[i]|0<=k*c[i]<=v}

如果将v的循环顺序从上面的0-1背包逆序改成顺序的话,那么则成了f[i][v]由f[i][v-c[i]]推知,刚好满足完全背包的定义可以将数组降维  

那么这里,我们顺序写,这里的max中的两项当然就是当前状态的值了,因为每种背包都是无限的。当我们把i从1到N循环时,

f[v]表示容量为v在前i种背包时所得的价值,这里我们要添加的不是前一个背包,而是当前背包。所以我们要考虑的当然是当前状态。

 

V=10,N=3,c[]={3,4,5}, w={4,5,6}

(1)背包不一定装满

  计算顺序是:从左往右,自上而下:  每个物品可以放多次,前面的会影响后面的

     

(2)背包刚好装满

  计算顺序是:从左往右,自上而下。注意初始值,其中-inf表示负无穷

     

代码如下:

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

这里同样给大家一道题目:

题目:http://acm.hdu.edu.cn/showproblem.php?pid=1114

代码:http://www.wutianqi.com/?p=535

(分析代码也是学习算法的一种途径,有时并不一定要看算法分析,结合题目反而更容易理解。)

一个简单有效的优化 
完全背包问题有一个很简单有效的优化,是这样的:若两件物品i、j满足c[i]<=c[j]且w[i]>=w[j],则将物品j去掉,不用考虑。这个优化的正确性显然:任何情况下都可将价值小费用高得j换成物美价廉的i,得到至少不会更差的方案。对于随机生成的数据,这个方法往往会大大减少物品的件数,从而加快速度。然而这个并不能改善最坏情况的复杂度,因为有可能特别设计的数据可以一件物品也去不掉。 

——————————————————————————————————————————————————————————–

 

三、多重背包

有N种物品和一个容量为V的背包。第i种物品最多有n[i]件可用,每件费用是c[i],价值是w[i]。求解将哪些物品装入背包可使这些物品的费用总和不超过背包容量,

且价值总和最大。

这题目和完全背包问题很类似。基本的方程只需将完全背包问题的方程略微一改即可,因为对于第i种物品有n[i]+1种策略:取0件,取1件……取n[i]件。令f[i][v]表示前i种物品恰放入一个容量为v的背包的最大权值,则有状态转移方程:

f[i][v]=max{f[i-1][v-k*c[i]]+k*w[i]|0<=k<=n[i]}

这里同样转换为01背包:

普通的转换对于数量较多时,则可能会超时

对于普通的。就是多了一个中间的循环,把j=0~bag[i],表示把第i中背包从取0件枚举到取bag[i]件。

给出一个例题:

题目:http://acm.hdu.edu.cn/showproblem.php?pid=2191

代码:http://www.wutianqi.com/?p=537

 

对于二进制办法

多重背包转换成 01 背包问题就是多了个初始化,把它的件数C 用二进制分解成若干个件数的集合,这里面数字可以组合成任意小于等于C的件数,而且不会重复,

之所以叫二进制分解,是因为这样分解可以用数字的二进制形式来解释  
       比如:7的二进制 7 = 111 它可以分解成 001 010 100 这三个数可以组合成任意小于等于7 的数,而且每种组合都会得到不同的数  
       15 = 1111 可分解成 0001  0010  0100  1000 四个数字  
       如果13 = 1101 则分解为 0001 0010 0100 0110 前三个数字可以组合成  7以内任意一个数,即1、2、4可以组合为1——7内所有的数,加上 0110 = 6 可以组合成任意一个大于6 小于等于13的数,比如12,可以让前面贡献6且后面也贡献6就行了。虽然有重复但总是能把 13 以内所有的数都考虑到了,基于这种思想去把多件物品转换为,多种一件物品,就可用01 背包求解了。

 

int n;  //输入有多少种物品
int c;  //每种物品有多少件
int v;  //每种物品的价值
int s;  //每种物品的尺寸
int count = 0; //分解后可得到多少种物品
int value[MAX]; //用来保存分解后的物品价值
int size[MAX];  //用来保存分解后物品体积

scanf("%d", &n);    //先输入有多少种物品,接下来对每种物品进行分解

while (n--)     //接下来输入n中这个物品
{
    scanf("%d%d%d", &c, &s, &v);  //输入每种物品的数目和价值
    for (int k=1; k<=c; k<<=1)   //<<右移 相当于乘二
    {
        value[count] = k*v;
        size[count++] = k*s;
        c -= k;
    }
    if (c > 0)
    {
        value[count] = c*v;
        size[count++] = c*s;
    }
}

 

定理:一个正整数n可以被分解成1,2,4,…,2^(k-1),n-2^k+1(k是满足n-2^k+1>0的最大整数)的形式,且1~n之内的所有整数均可以唯一表示成1,2,4,…,2^(k-1),n-2^k+1中某几个数的和的形式。

 

证明如下:

(1) 数列1,2,4,…,2^(k-1),n-2^k+1中所有元素的和为n,所以若干元素的和的范围为:[1, n];

(2)如果正整数t<= 2^k – 1,则t一定能用1,2,4,…,2^(k-1)中某几个数的和表示,这个很容易证明:我们把t的二进制表示写出来,很明显,t可以表示成n=a0*2^0+a1*2^1+…+ak*2^(k-1),其中ak=0或者1,表示t的第ak位二进制数为0或者1.

(3)如果t>=2^k,设s=n-2^k+1,则t-s<=2^k-1,因而t-s可以表示成1,2,4,…,2^(k-1)中某几个数的和的形式,进而t可以表示成1,2,4,…,2^(k-1),s中某几个数的和(加数中一定含有s)的形式。

(证毕!)

现在用count 代替 n 就和01 背包问题完全一样了  

杭电2191题解:http://acm.hdu.edu.cn/showproblem.php?pid=2191

此为多重背包用01和完全背包:

 1 #include<stdio.h>
 2 #include<string.h>
 3 int dp[102];
 4 int p[102],h[102],c[102];
 5 int n,m;
 6 void comback(int v,int w)//经费,重量。完全背包;
 7 {
 8     for(int i=v; i<=n; i++)
 9         if(dp[i]<dp[i-v]+w)
10             dp[i]=dp[i-v]+w;
11 }
12 void oneback(int v,int w)//经费,重量;01背包;
13 {
14     for(int i=n; i>=v; i--)
15         if(dp[i]<dp[i-v]+w)
16             dp[i]=dp[i-v]+w;
17 }
18 int main()
19 {
20     int ncase,i,j,k;
21     scanf("%d",&ncase);
22     while(ncase--)
23     {
24         memset(dp,0,sizeof(dp));
25         scanf("%d%d",&n,&m);//经费,种类;
26         for(i=1; i<=m; i++)
27         {
28             scanf("%d%d%d",&p[i],&h[i],&c[i]);//价值,重量,数量;
29             if(p[i]*c[i]>=n) comback(p[i],h[i]);
30             else
31             {
32                 for(j=1; j<c[i]; j<<1)
33                 {
34                     oneback(j*p[i],j*h[i]);
35                     c[i]=c[i]-j;
36                 }
37                 oneback(p[i]*c[i],h[i]*c[i]);
38             }
39         }
40         printf("%d\n",dp[n]);
41     }
42     return 0;
43 }

只是用01背包,用二进制优化:

 1 #include <iostream>
 2 using namespace std;
 3 int main()
 4 {
 5     int nCase,Limit,nKind,i,j,k,  v[111],w[111],c[111],dp[111];
 6     //v[]存价值,w[]存尺寸,c[]存件数
 7     //在本题中,价值是米的重量,尺寸是米的价格
 8     int count,Value[1111],size[1111];
 9     //count存储分解完后的物品总数
10     //Value存储分解完后每件物品的价值
11     //size存储分解完后每件物品的尺寸
12     cin>>nCase;
13     while(nCase--)
14     {
15         count=0;
16         cin>>Limit>>nKind;
17         for(i=0; i<nKind; i++)
18         {
19             cin>>w[i]>>v[i]>>c[i];
20             //对该种类的c[i]件物品进行二进制分解
21             for(j=1; j<=c[i]; j<<=1)
22             {
23                 //<<右移1位,相当于乘2
24                 Value[count]=j*v[i];
25                 size[count++]=j*w[i];
26                 c[i]-=j;
27             }
28             if(c[i]>0)
29             {
30                 Value[count]=c[i]*v[i];
31                 size[count++]=c[i]*w[i];
32             }
33         }
34         //经过上面对每一种物品的分解,
35         //现在Value[]存的就是分解后的物品价值
36         //size[]存的就是分解后的物品尺寸
37         //count就相当于原来的n
38         //下面就直接用01背包算法来解
39         memset(dp,0,sizeof(dp));
40         for(i=0; i<count; i++)
41             for(j=Limit; j>=size[i]; j--)
42                 if(dp[j]<dp[j-size[i]]+Value[i])
43                     dp[j]=dp[j-size[i]]+Value[i];
44 
45         cout<<dp[Limit]<<endl;
46     }
47     return 0;
48 }

未优化的

 1 #include <iostream>
 2 using namespace std;
 3 int main()
 4 {
 5     int nCase,Limit,nKind,i,j,k,  v[111],w[111],c[111],dp[111];
 6     //v[]存价值,w[]存尺寸,c[]存件数
 7     //在本题中,价值是米的重量,尺寸是米的价格
 8     int count,Value[1111],size[1111];
 9     //count存储分解完后的物品总数
10     //Value存储分解完后每件物品的价值
11     //size存储分解完后每件物品的尺寸
12     cin>>nCase;
13     while(nCase--)
14     {
15         count=0;
16         cin>>Limit>>nKind;
17         for(i=0; i<nKind; i++)
18         {
19             cin>>w[i]>>v[i]>>c[i];
20             //对该种类的c[i]件物品进行二进制分解
21             for(j=1; j<=c[i]; j<<=1)
22             {
23                 //<<右移1位,相当于乘2
24                 Value[count]=j*v[i];
25                 size[count++]=j*w[i];
26                 c[i]-=j;
27             }
28             if(c[i]>0)
29             {
30                 Value[count]=c[i]*v[i];
31                 size[count++]=c[i]*w[i];
32             }
33         }
34         //经过上面对每一种物品的分解,
35         //现在Value[]存的就是分解后的物品价值
36         //size[]存的就是分解后的物品尺寸
37         //count就相当于原来的n
38         //下面就直接用01背包算法来解
39         memset(dp,0,sizeof(dp));
40         for(i=0; i<count; i++)
41             for(j=Limit; j>=size[i]; j--)
42                 if(dp[j]<dp[j-size[i]]+Value[i])
43                     dp[j]=dp[j-size[i]]+Value[i];
44 
45         cout<<dp[Limit]<<endl;
46     }
47     return 0;
48 }

因为限于个人的能力,我只能讲出个大概,请大家具体还是好好看看dd大牛的《背包九讲》。

暂时讲完后,随着以后更深入的了解,我会把资料继续完善,供大家一起学习探讨。

本文下载地址(点击此处下载):Word版

posted @ 2015-06-14 11:54  繁夜  阅读(2707)  评论(0编辑  收藏  举报