基础背包问题
01背包
1:01背包
有 N 件物品和一个容量是 V 的背包。每件物品只能使用一次。
第 i 件物品的体积是 vi,价值是 wi。
求解将哪些物品装入背包,可使这些物品的总体积不超过背包容量,且总价值最大。
输出最大价值。
输入格式
第一行两个整数,N,V,用空格隔开,分别表示物品数量和背包容积。
接下来有 N 行,每行两个整数 vi,wi,用空格隔开,分别表示第 i 件物品的体积和价值。
输出格式
输出一个整数,表示最大价值。
数据范围
0<N,V≤1000
0<vi,wi≤1000
输入样例
4 5
1 2
2 4
3 4
4 5
输出样例:
8
思路:
题意为在n件物品中挑选体积不超过m的物品,使之总价值最大。我们令dp[i][j]表示有i件物品体积不超过j时的最大价值。那么对于每件物品 ,我们有选或者不选两种状态 。每次我们取最大价值的拿法,最终输出dp[n][m]即可。状态转移方程为dp[i][j]=max(dp[i-1][j],dp[i-1][j-v]+w) 。这里需要注意的是 ,动态规划中后面的每一项 ,都是由前一项转移而来 。我们可以对其空间进行优化 ,将二维数组改为一维数组 ,以dp[j]表示有i件物品体积不超过j时的最大价值 。此时在枚举体积时,我们需要从大到小枚举。因为我们在求i件物品时 ,需要的利用是i-1件物品时的数据。假如我们从小到大枚举 ,那么在求第i件物品 j体积时小于j的数值已经被我们更新为第i件时的数据了,导致我们无法进行后面的计算 。还需要注意的一个细节是 ,求体积恰好是m时最大价值和体积最大是m时的最大价值有什么区别呢? 区别就在于dp[j]数组的初始化不同。
1:当我们求的是体积恰好为m时的最大价值 ,我们需要把 dp[0] 初始化为 0 ,而dp[i] (1<=i<=m)初始化为负无穷 。为什么呢?因为此时dp[j]表示有0件物品且背包容量为j时的最大价值 。0件物品 ,只有当背包容量为0时 ,才能恰好装满背包,此时最大价值为0。负无穷表示其他条件下有 0件物品背包容量大于0时背包不可能装满 。
2:当我们求的是体积不超过j时的最大价值 ,那么我们将dp[j] (j>=0)都初始化为0 。表示在有 0件物品且不要求装满的情况下 ,无论如何最大价值都是 0 。
#include <stdio.h>
int n,m,a,b,dp[1005];
int max(int x,int y)
{
return x>y?x:y;
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=0;i<n;i++)
{
scanf("%d%d",&a,&b);
for(int j=m;j>=a;j--)
{
dp[j]=max(dp[j],dp[j-a]+b);
}
}
printf("%d",dp[m]);
return 0;
}
2: 采药
辰辰是个天资聪颖的孩子,他的梦想是成为世界上最伟大的医师。
为此,他想拜附近最有威望的医师为师。
医师为了判断他的资质,给他出了一个难题。
医师把他带到一个到处都是草药的山洞里对他说:“孩子,这个山洞里有一些不同的草药,采每一株都需要一些时间,每一株也有它自身的价值。我会给你一段时间,在这段时间里,你可以采到一些草药。如果你是一个聪明的孩子,你应该可以让采到的草药的总价值最大。”
如果你是辰辰,你能完成这个任务吗?
输入格式
输入文件的第一行有两个整数T和M,用一个空格隔开,T代表总共能够用来采药的时间,M代表山洞里的草药的数目。
接下来的M行每行包括两个在1到100之间(包括1和100)的整数,分别表示采摘某株草药的时间和这株草药的价值。
输出格式
输出文件包括一行,这一行只包含一个整数,表示在规定的时间内,可以采到的草药的最大总价值。
数据范围
1≤T≤1000,
1≤M≤100
输入样例:
70 3
71 100
69 1
1 2
输出样例:
3
#include <stdio.h>
int n,m,a,b,dp[1005];
int max(int x,int y)
{
return x>y?x:y;
}
int main()
{
scanf("%d%d",&m,&n);
for(int i=0;i<n;i++)
{
scanf("%d%d",&a,&b);
for(int j=m;j>=a;j--)
{
dp[j]=max(dp[j],dp[j-a]+b);
}
}
printf("%d",dp[m]);
return 0;
}
完全背包
1:完全背包问题
有 N 种物品和一个容量是 V 的背包,每种物品都有无限件可用。
第 i 种物品的体积是 vi,价值是 wi。
求解将哪些物品装入背包,可使这些物品的总体积不超过背包容量,且总价值最大。
输出最大价值。
输入格式
第一行两个整数,N,V,用空格隔开,分别表示物品种数和背包容积。
接下来有 N 行,每行两个整数 vi,wi,用空格隔开,分别表示第 i 种物品的体积和价值。
输出格式
输出一个整数,表示最大价值。
数据范围
0<N,V≤1000
0<vi,wi≤1000
输入样例
4 5
1 2
2 4
3 4
4 5
输出样例:
10**
**思路:**完全背包和01背包的区别就是每件物品可以取无数次 ,类比01背包 ,很容易的我们便得到了状态转移方程 dp[i][j]=max(dp[i-1][j],dp[i-1][j -k * v ]+k * w)。 和01背包一样 ,这里我们也可以进行空间 上的优化 。下面的代码已经十分简洁了,仔细想想为什么可以这样写。。。。。。因为每件物品可以取无数次,所以可以从小向大写。
#include <stdio.h>
int dp[1001],v,w,n,m;
int max(int x,int y)
{
return x>y?x:y;
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=0;i<n;i++)
{
scanf("%d%d",&v,&w);
for(int j=v;j<=m;j++)
{
dp[j]=max(dp[j],dp[j-v]+w);
}
}
printf("%d",dp[m]);
return 0;
}
多重背包
1 :多重背包问题
有 N 种物品和一个容量是 V 的背包。
第 i 种物品最多有 si 件,每件体积是 vi,价值是 wi。
求解将哪些物品装入背包,可使物品体积总和不超过背包容量,且价值总和最大。
输出最大价值。
输入格式
第一行两个整数,N,V,用空格隔开,分别表示物品种数和背包容积。
接下来有 N 行,每行三个整数 vi,wi,si,用空格隔开,分别表示第 i 种物品的体积、价值和数量。
输出格式
输出一个整数,表示最大价值。
数据范围
0<N,V≤100
0<vi,wi,si≤100
输入样例
4 5
1 2 3
2 4 1
3 4 3
4 5 2
输出样例:
10
**思路:**多重背包的特点就是每件物品有确定的数量 ,我们用dp[i][j]表示有i件物品体积最大是j时的最大价值 。状态转移方程为dp[i][j]=max(dp[i-1][j−c[i]∗k]+w[i]∗k,dp[i-1][j]); 同理 ,我们先可以对其进行空间上的优化·,以一维数组的形式储存数据。
第一份代码如下:
#include <stdio.h>
int dp[1001],n,m,v,w,s;
int max(int x,int y)
{
return x>y?x:y;
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=0;i<n;i++)
{
scanf("%d%d%d",&v,&w,&s);
for(int j=m;j>=v;j--)
{
for(int k=1;k*v<=j&&k<=s;k++)
{
dp[j]=max(dp[j],dp[j-k*v]+k*w);
}
}
}
printf("%d",dp[m]);
return 0;
}
上面的代码可以解决一定数据范围的多重背包问题 。但是时间复杂度过高 。在这里我们还可以采用二进制优化 。即一个正整数 n,可以f分解为(1 ,2 ,4, 8·····最大数)的形式 。什么意思呢?举个例子 :n的值为10 ,那么10可以分解为 1 ,2 ,4,3 。为什么这样分解呢?因为这样一来 ,10以内的所有正整数均能用我们分解出来的数表示出来 。这样 ,我们便可以将多重背包转变为01背包,对不对,是不是很巧妙。
二进制优化的多重背包:
#include <stdio.h>
int dp[100000],n,m,v,w,s,t,k=0;
int max(int x,int y)
{
return x>y?x:y;
}
struct dalao
{
int V;
int W;
}p[100000];
int main()
{
scanf("%d%d",&n,&m);
for(int i=0;i<n;i++)
{
scanf("%d%d%d",&v,&w,&s);
t=1;
while(t<=s)
{
s-=t;
p[k].V=t*v;
p[k].W=t*w;
t*=2;
k++;
}
if(s>0)
{
p[k].V=s*v;
p[k].W=s*w;
k++;
}
}
for(int i=0;i<k;i++)
{
for(int j=m;j>=p[i].V;j--)
{
dp[j]=max(dp[j],dp[j-p[i].V]+p[i].W);
}
}
printf("%d",dp[m]);
return 0;
}
上面的代码很好 ,但是多重背包还有更好的优化方法 。即用单调队列优化 。但是由于本文篇幅有限 ,有兴趣的可以自行百度 。hahahah``。。。
混合背包
1:混合背包问题
有 N 种物品和一个容量是 V 的背包。
物品一共有三类:
第一类物品只能用1次(01背包);
第二类物品可以用无限次(完全背包);
第三类物品最多只能用 si 次(多重背包);
每种体积是 vi,价值是 wi。
求解将哪些物品装入背包,可使物品体积总和不超过背包容量,且价值总和最大。
输出最大价值。
输入格式
第一行两个整数,N,V,用空格隔开,分别表示物品种数和背包容积。
接下来有 N 行,每行三个整数 vi,wi,si,用空格隔开,分别表示第 i 种物品的体积、价值和数量。
si=−1 表示第 i 种物品只能用1次;
si=0 表示第 i 种物品可以用无限次;
si>0 表示第 i 种物品可以使用 si 次;
输出格式
输出一个整数,表示最大价值。
数据范围
0<N,V≤1000
0<vi,wi≤1000
−1≤si≤1000
输入样例
4 5
1 2 -1
2 4 1
3 4 0
4 5 2
输出样例:
8
**思路:**混合背包其实就是上面三种背包的混合 。将多重背包转化为01背包 ,分为01背包和完全背包两种情况求解。
#include <stdio.h>
int dp[100000],v,w,n,m,s,t,k=0;
int max(int x,int y)
{
return x>y?x:y;
}
struct dalao
{
int V;
int W;
int S;
}p[100000];
int main()
{
scanf("%d%d",&n,&m);
for(int i=0;i<n;i++)
{
scanf("%d%d%d",&v,&w,&s);
if(s>0)
{
t=1;
while(t<=s)
{
s-=t;
p[k].V=v*t;
p[k].W=w*t;
p[k].S=-1;
k++;
t*=2;
}
if(s>0)
{
p[k].V=v*s;
p[k].W=w*s;
p[k].S=-1;
k++;
}
}
else
{
p[k].V=v;
p[k].W=w;
p[k].S=s;
k++;
}
}
for(int i=0;i<k;i++)
{
if(p[i].S==0)
{
for(int j=p[i].V;j<=m;j++)
{
dp[j]=max(dp[j],dp[j-p[i].V]+p[i].W);
}
}
else
{
for(int j=m;j>=p[i].V;j--)
{
dp[j]=max(dp[j],dp[j-p[i].V]+p[i].W);
}
}
}
printf("%d",dp[m]);
return 0;
}
分组背包
1:分组背包问题
有 N 组物品和一个容量是 V 的背包。
每组物品有若干个,同一组内的物品最多只能选一个。
每件物品的体积是 vij,价值是 wij,其中 i 是组号,j 是组内编号。
求解将哪些物品装入背包,可使物品总体积不超过背包容量,且总价值最大。
输出最大价值。
输入格式
第一行有两个整数 N,V,用空格隔开,分别表示物品组数和背包容量。
接下来有 N 组数据:
每组数据第一行有一个整数 Si,表示第 i 个物品组的物品数量;
每组数据接下来有 Si 行,每行有两个整数 vij,wij,用空格隔开,分别表示第 i 个物品组的第 j 个物品的体积和价值;
输出格式
输出一个整数,表示最大价值。
数据范围
0<N,V≤100
0<Si≤100
0<vij,wij≤100
输入样例
3 5
2
1 2
2 4
1
3 4
1
4 5
输出样例:
8
思路: 类比于01背包,01背包有n件物品 ,每件物品有2个选择 ,即选或者不选 。我们可以把分组背包看成01背包 ,每组看成一个物品 ,对应的每组有s+1种选择 。第i组的每种选择都是建立在第i-1次的数据上。
#include <stdio.h>
int n,m,v[105],w[105],s,dp[105];
int max(int x,int y)
{
return x>y?x:y;
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=0;i<n;i++)
{
scanf("%d",&s);
for(int i=0;i<s;i++)
{
scanf("%d%d",&v[i],&w[i]); //储存本组每种决策的代价和价值
}
for(int j=m;j>=0;j--) //对于本组的每一个决策 都是基于上一组的数据 所以从大到小
{
for(int k=0;k<s;k++) //循环本组的物品 同组对于本组同一重量间竞争 每个重量 对应于本组的一个或0个物品
{
if(v[k]<=j) //如果数据合法
{
dp[j]=max(dp[j],dp[j-v[k]]+w[k]);
}
}
}
}
printf("%d",dp[m]);
return 0;
}
**
二维费用的背包问题
**
有 N 件物品和一个容量是 V 的背包,背包能承受的最大重量是 M。
每件物品只能用一次。体积是 vi,重量是 mi,价值是 wi。
求解将哪些物品装入背包,可使物品总体积不超过背包容量,总重量不超过背包可承受的最大重量,且价值总和最大。
输出最大价值。
输入格式
第一行两个整数,N,V,M,用空格隔开,分别表示物品件数、背包容积和背包可承受的最大重量。
接下来有 N 行,每行三个整数 vi,mi,wi,用空格隔开,分别表示第 i 件物品的体积、重量和价值。
输出格式
输出一个整数,表示最大价值。
数据范围
0<N≤1000
0<V,M≤100
0<vi,mi≤100
0<wi≤1000
输入样例
4 5 6
1 2 3
2 4 4
3 4 5
4 5 6
输出样例:
8
思路 : 其实就是在01背包的基础上 ,加了一层循环 ,写法和01背包一样。
#include <stdio.h>
int n,m,v,dp[1005][1005],a,b,c;
int max(int x,int y)
{
return x>y?x:y;
}
int main()
{
scanf("%d%d%d",&n,&v,&m); //件数 体积 重量
for(int i=0;i<n;i++)
{
scanf("%d%d%d",&a,&b,&c); //体积 重量 价值
for(int j=v;j>=a;j--)
{
for(int k=m;k>=b;k--)
{
dp[j][k]=max(dp[j][k],dp[j-a][k-b]+c);
}
}
}
printf("%d",dp[v][m]);
return 0;
}
背包问题求方案数
有 N 件物品和一个容量是 V 的背包。每件物品只能使用一次。
第 i 件物品的体积是 vi,价值是 wi。
求解将哪些物品装入背包,可使这些物品的总体积不超过背包容量,且总价值最大。
输出 最优选法的方案数。注意答案可能很大,请输出答案模 109+7 的结果。
输入格式
第一行两个整数,N,V,用空格隔开,分别表示物品数量和背包容积。
接下来有 N 行,每行两个整数 vi,wi,用空格隔开,分别表示第 i 件物品的体积和价值。
输出格式
输出一个整数,表示 方案数 模 109+7 的结果。
数据范围
0<N,V≤1000
0<vi,wi≤1000
输入样例
4 5
1 2
2 4
3 4
4 601
输出样例:
2
思路: 因为我们要求方案数 ,为了方便记录 。我们用dp1[i]表示前i件物品体积恰好为j时的最大价值 。与01背包不同的是 ,我们需要另外开一个数组记录体积恰好为i时最大价值对应的方案数。对于每个物品 ,我们判断在体积恰好为j是否选择他,分为选他、不选他、选或者不选均是最大价值三种情况 。
#include <stdio.h>
int n,m,a,b,dp1[1005],dp2[1005],mod=1e9+7,s;
int max(int x,int y)
{
return x>y?x:y;
}
int main() //dp1[i]表示体积恰好是i时的最大价值 dp2[i]表示体积恰好为i时最大价值对应的方案数
{
scanf("%d%d",&n,&m);
for(int i=1;i<=m;i++)
dp1[i]=-1e9;
dp2[0]=1;
for(int i=0;i<n;i++)
{
scanf("%d%d",&a,&b); //体积 价值
for(int j=m;j>=a;j--)
{
s=0;
int t=max(dp1[j],dp1[j-a]+b);
if(t==dp1[j]) s+=dp2[j]; //判断那种方案价值最大 (有可能一样大)
if(t==dp1[j-a]+b) s+=dp2[j-a];
dp1[j]=t;
dp2[j]=s%mod;
}
}
int MAX=-1,ans=0;
for(int i=0;i<=m;i++)
{
MAX=max(dp1[i],MAX);
}
for(int i=0;i<=m;i++)
{
if(MAX==dp1[i])
ans+=dp2[i];
}
printf("%d",ans);
return 0;
}