题目描述
算法思路分析
我在以前的一篇博客文章中写过如何求解简单的0-1背包问题,部分背包问题我觉得要比简单的0-1背包问题简单,他们两个所采用的都是贪心的策略,总是做出对当前最好的选择,与简单的0-1背包不同的是,部分背包问题可以只装一部分,这样的话实现思路是这样的:我们求出每一堆金币的单位质量的价值per-w[i],然后将每一堆金币按照单位质量的金币价值进行降序排序,尽量将单位质量价值大的金币装进背包,这样就可以使得背包可以装进的所有金币的价值最大,想了解简单0-1背包的可以去看我上一篇博客,博客链接:http://t.csdn.cn/DO6tN
算法源代码以及运行结果截图
算法源代码:
//部分背包问题
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#define N 100
int value[N] = { 0 }, weight[N] = { 0 };
double per_w[N] = { 0 };
int main()
{
int n, t, now_w; //n表示有n堆金币,而t表示阿里巴巴的背包所能承载的重量,now_w表示背包中现在已有的金币的重量;
double total_v; //表示背包可以装的金币的总价值
while (scanf("%d%d", &n, &t) != EOF)
{
now_w = total_v = 0;
for (int i = 0; i < n; i++)
{
scanf("%d%d", &weight[i], &value[i]);
per_w[i] = (double)value[i] / weight[i];
}
//按照单位质量的价值大小进行降序排序
for (int i = 0; i < n - 1; i++)
{
for (int j = 0; j < n - 1 - i; j++)
{
if (per_w[j] < per_w[j + 1])
{
double temp1;
temp1 = per_w[j];
per_w[j] = per_w[j + 1];
per_w[j + 1] = temp1;
int temp2;
temp2 = weight[j];
weight[j] = weight[j + 1];
weight[j + 1] = temp2;
temp2 = value[j];
value[j] = value[j + 1];
value[j + 1] = temp2;
}
}
}
//将金币装进背包,每次总是装单位质量价值最大的金币
for (int i = 0; i < n; i++)
{
if (weight[i] <= t - now_w) //当某一堆金币的重量小于背包当前还能再承载的重量的时候就将该金币全部装进背包
{
total_v += value[i];
now_w += weight[i];
}
else
{
total_v = total_v + (t - now_w) * per_w[i]; //当该堆金币的重量超过背包目前还能承载的重量的时候就只能装该堆金币的部分,直到将背包装满为止
break;
}
}
printf("%.2lf\n", total_v);
}
}
运行结果截图: