问题:给定n种物品和一个背包。物品i的重量是W,其价值为V,背包最大承载重量为C。物品是可分割的,请选择装入背包的物品,使装入背包物品总价值最大。请分别编写递推(循环迭代)和递归程序求解
输入:随机生成N个物品的基本数据和自定义背包容量C。
输出:背包能装下的最大价值和所选择物品的编号及重量。
该问题是典型的贪心算法问题,贪心最大价值,约束条件为背包载重上限。首先求出物品单价后进行从大到小的排序然后,然后选取单价最高的物品依次放入背包。
代码:
#include<stdlib.h>
#include<time.h>
#include<stdio.h>
const int N=10;
struct goods{
int No;//编号
int w;//重量
int v;//价值
double avg;//单位重量的价值
};
void selectsort(goods g[]){//选择排序,将物品单价进行排序
for(int i=0;i<N;i++)
for(int j=i;j<N;j++)
if(g[j].avg>g[i].avg){
goods x;
x=g[i];
g[i]=g[j];
g[j]=x;
}
}
//******************************************递推算法***************************
double greedy_tui(goods g[],int c)
{
double value=0;
for(int i=0;i<N;i++){
if(c>g[i].w)//当背包剩余重量大于物品重量时
{ c=c-g[i].w;//剩余容量减去物品重量,求加上物品后背包剩余重量
value=value+g[i].v; }
else if(c<g[i].w)//当背包剩余重量小于物品重量时
{ value=value+c*g[i].avg;//将物品拆分,求背包装满时的总价值
break; }
else{//当背包刚好装满时
value=value+g[i].v;
break; }}
//对排好序的物品按顺序进行贪心 ,
return value;
}
//******************************************递归算法********************************
double value1=0;
double greedy_gui(goods g[],int c,int i)
{
if(c-g[i].w==0){//刚好装满
value1=value1+g[i].v;
printf("背包恰好装满啦\n");
return value1;
}
else if(c-g[i].w<0){//重量溢出
value1=value1+c*g[i].avg;
printf("需要切割,切出重量为%d的物品%d装入背包\n",c,g[i].No);
return value1;//
}
else{//背包未装满,向下一层递归
value1=value1+g[i].v;
printf("可以将物品%d全部装入背包\n",g[i].No);
return greedy_gui(g,c-g[i].w,i+1);
}
}
//*********************************主函数*******************************
int main()
{
int c;
printf("请输入背包最大承重(整数):");
scanf("%d",&c);
goods g[N];
for(int i=0;i<N;i++)//随机生成物品参数
{
g[i].No=i;
g[i].w=rand()%10+1;
g[i].v=rand()%10+1;
g[i].avg=g[i].v;
g[i].avg=g[i].avg/g[i].w;
printf("物品%d重%d公斤,价值%d元\n",g[i].No,g[i].w,g[i].v);
}
selectsort(g);
printf("排序后:\n");
for(int i=0;i<N;i++) printf("物品%d重%d公斤,价值%d元\n",g[i].No,g[i].w,g[i].v);
printf("递推能装入背包的最大价值为%f元\n",greedy_tui(g,c));
printf("递归能装入背包的最大价值为%f元\n",greedy_gui(g,c,0));
return 0;
}