背包问题

1 篇文章 0 订阅

首先是0-1背包问题,这是最基本的背包问题,给定一个背包,容量为V,给定n个物品,并给出每个物品的体积和价值,每个物体只有一件。

下面是0-1背包问题的代码模型:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define Max 110
#define Maxv 1010
#define Maxx(a,b) (a)>(b)?(a):(b)
int weight[Max];
int value[Max];
int dp[Maxv];
int V;
int n;
int main(){
	scanf("%d",&n);
	for(int i=1;i<=n;i++)
		scanf("%d%d",&weight[i],&value[i]);
	memset(dp,0,sizeof(dp));
	for(int i=1;i<=n;i++)
		for(int j=V;j>=weight[i];j--)
			dp[j]=Maxx(dp[j],dp[j-weight[i]]+value[i]);
        printf("%d\n",dp[V]);
	return 0;
}

完全背包问题:
区别与0-1背包问题的关键点是:物体的个数是无穷,下面是代码模型:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define Max 110
#define Maxv 1010
#define Maxx(a,b) (a)>(b)?(a):(b)
#define Inf 11111100
int weight[Max];
int value[Max];
int dp[Maxv];
int V;
int n;
int main(){
	scanf("%d",&n);
	for(int i=1;i<=n;i++)
		scanf("%d%d",&weight[i],&value[i]);
	memset(dp,0,sizeof(dp));
	//要求刚好装满
	//for(int i=1;i<=V;i++)
	//dp[i]=-Inf;
	//dp[0]=0;
	for(int i=1;i<=n;i++)
		for(int j=weight[i];j<=V;j++)
			dp[j]=Maxx(dp[j],dp[j-weight[i]]+value[i]);
	printf("%d\n",dp[V]);
	return 0;
}

多重背包问题:

区别于0-1背包问题的关键在于物体的个数为num[i]下面是没有优化的代码模型

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define Max 110
#define Maxv 1010
#define Maxx(a,b) (a)>(b)?(a):(b)
#define Inf 11111100
int weight[Max];
int value[Max];
int num[Max];
int dp[Maxv];
int V;
int n;
int main(){
	scanf("%d",&n);
	for(int i=1;i<=n;i++)
		scanf("%d%d%d",&weight[i],&value[i],&num[i]);
	memset(dp,0,sizeof(dp));
	//要求刚好装满
	//for(int i=1;i<=V;i++)
	//dp[i]=-Inf;
	//dp[0]=0;
	for(int i=1;i<=n;i++)
		for(int k=0;k<=num[i];k++)
		  for(int j=V;j>=weight[i];j--)
			dp[j]=Maxx(dp[j],dp[j-weight[i]]+value[i]);
	printf("%d\n",dp[V]);
	return 0;
}
以hdu2191为例说明:
下面是优化代码一:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define Max(a,b) (a)>(b)?(a):(b)
#define Maxv 110
#define Maxx 20010
int V,n,Case;
int dp[Maxv];
int weight[Maxx];
int value[Maxx];
int main(){
    scanf("%d",&Case);
    while(Case--){
        scanf("%d%d",&V,&n);
        memset(dp,0,sizeof(dp));
        int pivot=0;
        int ww,vv,nn;
        for(int i=1;i<=n;i++){
            scanf("%d%d%d",&ww,&vv,&nn);
            for(int j=1;j<nn;j<<=1){
                weight[pivot]=j*ww;
                value[pivot++]=j*vv;
                nn-=j;
            }
            weight[pivot]=nn*ww;
            value[pivot++]=nn*vv;
        }
        for(int i=0;i<pivot;i++)
            for(int j=V;j>=weight[i];j--)
                dp[j]=Max(dp[j],dp[j-weight[i]]+value[i]);
        printf("%d\n",dp[V]);
    }
    return 0;
}

 

下面是优化代码二:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define Maxv 110
#define Max(a,b) (a)>(b)?(a):(b)
int dp[Maxv];
int n,V,Case;
int ww,vv,nn;
void onepage(int weight,int value){
    for(int i=V;i>=weight;i--)
        dp[i]=Max(dp[i],dp[i-weight]+value);
}
void compage(int weight,int value){
    for(int i=weight;i<=V;i++)
        dp[i]=Max(dp[i],dp[i-weight]+value);
}
int main(){
    scanf("%d",&Case);
    while(Case--){
        scanf("%d%d",&V,&n);
        memset(dp,0,sizeof(dp));
        for(int i=1;i<=n;i++){
           scanf("%d%d%d",&ww,&vv,&nn);
           if(ww*nn>=V)
               compage(ww,vv);
           else{
               for(int j=1;j<nn;j<<=1){
                   onepage(ww*j,vv*j);
                   nn-=j;
               }
               onepage(nn*ww,nn*vv);
           }
        }
        printf("%d\n",dp[V]);
    }
    return 0;
}

背包问题的核心是0-1背包问题的动态转移方程。dp[v]=max{dp[v],dp[v-weight]+value}

另外就是注意初始化问题。


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值