NOI / 4.6算法之贪心1797:金银岛(详细讲解)

4 篇文章 0 订阅

总时间限制: 3000ms                内存限制: 65536kB

描述

某天KID利用飞行器飞到了一个金银岛上,上面有许多珍贵的金属,KID虽然更喜欢各种宝石的艺术品,可是也不拒绝这样珍贵的金属。但是他只带着一个口袋,口袋至多只能装重量为w的物品。岛上金属有s个种类, 每种金属重量不同,分别为n1, n2, ... , ns,同时每个种类的金属总的价值也不同,分别为v1,v2, ..., vs。KID想一次带走价值尽可能多的金属,问他最多能带走价值多少的金属。注意到金属是可以被任意分割的,并且金属的价值和其重量成正比。

输入

第1行是测试数据的组数k,后面跟着k组输入。

每组测试数据占3行,第1行是一个正整数w (1 <= w <= 10000),表示口袋承重上限。第2行是一个正整数s (1 <= s <=100),表示金属种类。第3行有2s个正整数,分别为n1, v1, n2, v2, ... , ns, vs分别为第一种,第二种,...,第s种金属的总重量和总价值(1 <= ni <= 10000, 1 <= vi <= 10000)。

输出

k行,每行输出对应一个输入。输出应精确到小数点后2位。

样例输入

2
50
4
10 100 50 30 7 34 87 100
10000
5
1 43 43 323 35 45 43 54 87 43

样例输出 

171.93
508.00

 查看          提交          统计          提问

———————————————————————————————————————————

 开始讲解:

首先观察题目,每个金属都应有2个成员:重量和总价。但是我们发现金属可以切割,所以可能有不完整的金属,所以还要定义一个单价,而且单价必须是浮点数类型(重量和总价可以不是,但单价必须是浮点数),边输入一种金属的重量和总价,边计算单价。

定义结构体

struct node//结构体放金属的数据
{
	int n,v;//重量和总价值
	double m;//m表示单价
}a[102];最多100种金属

输入 

cin>>w>>s;//口袋承重及金属种类
for(i=0;i<s;i++)
{
	cin>>a[i].n>>a[i].v;//输入金属重量及总价值
	a[i].m=a[i].v*1.0/a[i].n;//计算单价
}

然后继续审题,KID飞到了一个金银岛上,上面有许多珍贵的金属,但是他只带着一个口袋,问他最多能带走价值多少的金属。属于典型的贪心策略,最多拿走多少金属,肯定是要先拿最贵的,拿完之后那剩下中的最贵的,以此类推。所以我们先给结构体数组按单价的降序排个序。

sort(a,a+s,cmp);
//cmp表示自己定义的结构体排序规则

cmp里内容

bool cmp(node a,node b)//a表示前一个结构体数组元素,b表示后一个结构体数组元素
{
	return a.m>b.m;//按照每个结构体数组元素中的m(单价)进行降序
}

cmp具体用法可见博主的其他文章C++中的cmp讲解 

接下来,我们需要模拟一下取金属的过程

for(i=0;i<s;i++)//优先拿最贵的,最后拿最便宜的。
if(w>=a[i].n)//如果口袋的容量大于重量。
{
	w-=a[i].n;//全部装走。
	ans+=a[i].v;//加上这种金属的总价值
}
else
{
	ans+=w*a[i].m;//能拿多少拿多少。
	w=0;
	break;//提前退出。
}

最后输出取走的总价值就行了,保留两位小数得用printf

printf("%.2f\n",ans);//输出总价值

呈现代码 

#include<bits/stdc++.h>
using namespace std;
struct node//结构体数组分别存金属的重量,总价,单价。
{
	int n,v;
	double m;
}a[105];//最多100种金属。
bool cmp(node a,node b)//前一组结构体数组元素和后一组结构体数组元素。
{
	return a.m>b.m;//排列规则为按单价降序排列。
}
int main()
{
	int k,i,w,s;
	double ans;//最终的总价值是小数。
	cin>>k;
	while(k--)//K组数
	{
		ans=0;//刚开始的价值为零。
		cin>>w>>s;//口袋的容量和金属的种类。
		for(i=0;i<s;i++)
		{
			cin>>a[i].n>>a[i].v;//输入重量和总价。
			a[i].m=a[i].v*1.0/a[i].n;//计算单价
		}
		sort(a,a+s,cmp);//按单价的降序排列。
		for(i=0;i<s;i++)//优先拿最贵的,最后拿最便宜的。
		if(w>=a[i].n)//如果口袋的容量大于重量。
		{
			w-=a[i].n;//全部装走。
			ans+=a[i].v;//加上这种金属的总价值
		}
		else
		{
			ans+=w*a[i].m;//能拿多少拿多少。
			w=0;
			break;//提前退出。
		}
		printf("%.2f\n",ans);//输出总价值
	}
	return 0;
}

 有不懂的问题可以留在评论区,我会在24小时内回复(✪ω✪)

  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值