USACO Section 1.1 Greedy Gift Givers 解题报告
题目
问题描述
有若干个朋友,朋友之间可以选择互相赠送一些有价值的礼物。一个人可以选择将一部分钱分给若干个朋友,例如某人送给其他两个人钱,总共赠送3元,两个人平均分,原本应该是每人1.5元,但是只能取整数,所以这两个人每人能够得到1元,所以他总共送出去2元,自己剩下1元。
现在,我们知道有NP个人,将NP个人的名字告诉我们,随后给出一个人的名字代表他将进行捐赠,下一行有两个数字,代表这个人的初始的金额X,和要将X元平分给Y个人,下面Y行分别是接受捐赠的人的名字。
最后需要我们计算出,每个人最终的金额与初始金额的差,可能会有负数出现。
数据范围
- 名字不会超过14个字符
2 <= NP <= 10
0 <= X <= 2000
样例输入
5
dave
laura
owen
vick
amr
dave
200 3
laura
owen
vick
owen
500 1
dave
amr
150 2
vick
owen
laura
0 2
amr
vick
vick
0 0
样例输出
dave 302
laura 66
owen -359
vick 141
amr -150
解题思路
定义一个Person结构体,将姓名、初始金额,和实时金额作为属性保存。之后开始模拟操作即可,在进行除法时请注意避免除数为0的情况。
解题代码
/*
ID: yinzong2
PROG: gift1
LANG: C++11
*/
#define MARK
#include<cstdio>
#include<iostream>
#include<string>
#include<cstring>
using namespace std;
const int maxn = 20;
int num;
struct Person {
char name[maxn];
int iniMoney;
int money;
} p[maxn];
int main() {
#ifdef MARK
freopen("gift1.in", "r", stdin);
freopen("gift1.out", "w", stdout);
#endif
while(~scanf("%d", &num)) {
for(int i = 0; i < num; i++) {
scanf("%s", p[i].name);
p[i].iniMoney = p[i].money = 0;
}
char str1[maxn], str2[maxn];
int ini, cnt;
for(int i = 0; i < num; i++) {
scanf("%s", str1);
scanf("%d%d", &ini, &cnt);
for(int j = 0; j < num; j++) {
if(strcmp(str1, p[j].name) == 0) {
p[j].iniMoney = ini;
p[j].money += ini;
if(cnt != 0)p[j].money -= (ini/cnt)*cnt;
break;
}
}
for(int j = 0; j < cnt; j++) {
scanf("%s", str2);
for(int k = 0; k < num; k++) {
if(strcmp(str2, p[k].name) == 0) {
if(cnt != 0)p[k].money += (ini/cnt);
break;
}
}
}
}
for(int i = 0; i < num; i++) {
printf("%s %d\n", p[i].name, p[i].money-p[i].iniMoney);
}
}
return 0;
}
代码优化
在上一个代码中,每次出现一个人名我必须用循环O(N)
的复杂度来查找,确定这个人名是谁之后才能进行操作。现在我用一个map来保存Person的信息,其中名字作为查找的key,对应这个人出现的次序作为value。在代码中我用了一个Person数组p来存储这些人的信息,现在这样用map处理之后,每次得到一个人名,我都能利用p[ map[name] ]
这种语句来得到这个人,map查找的时间复杂度为O(logN)
,如果数据量很大的话这种优化还是挺有效率的。
解题代码(Type 2)
/*
ID: yinzong2
PROG: gift1
LANG: C++11
*/
#define MARK
#include<cstdio>
#include<cstring>
#include<iostream>
#include<string>
#include<map>
using namespace std;
const int maxn = 20;
struct Person {
string name;
int iniMoney;
int money;
}p[maxn];
map<string, int> mp;
int np;
int main() {
#ifdef MARK
freopen("gift1.in", "r", stdin);
freopen("gift1.out", "w", stdout);
#endif
while(~scanf("%d", &np)) {
string str;
for(int i = 0; i < np; i++) {
cin >> str;
p[i].name = str;
p[i].iniMoney = p[i].money = 0;
mp[str] = i;
}
int ini, cnt;
for(int i = 0; i < np; i++) {
cin >> str;
scanf("%d%d", &ini, &cnt);
int id = mp[str];
p[id].iniMoney = ini;
p[id].money += ini;
if(cnt != 0) {
p[id].money -= (ini/cnt)*cnt;
}
for(int j = 0; j < cnt; j++) {
cin >> str;
int id2 = mp[str];
if(cnt != 0) {
p[id2].money += (ini/cnt);
}
}
}
for(int i = 0; i < np; i++) {
cout << p[i].name << " " << (p[i].money-p[i].iniMoney) << endl;
}
}
return 0;
}
USACO Section 1.1 Greedy Gift Givers 解题报告的更多相关文章
- USACO Section 1.1-2 Greedy Gift Givers
Greedy Gift Givers 贪婪的送礼者 对于一群(NP个)要互送礼物的朋友,GY要确定每个人送出的钱比收到的多多少. 在这一个问题中,每个人都准备了一些钱来送礼物,而这些钱将会被平均分给那 ...
- Section 1.1 Greedy Gift Givers
Greedy Gift Givers A group of NP (2 ≤ NP ≤ 10) uniquely named friends hasdecided to exchange gifts o ...
- usaco training <1.2 Greedy Gift Givers>
题面 Task 'gift1': Greedy Gift Givers A group of NP (2 ≤ NP ≤ 10) uniquely named friends has decided t ...
- USACO 1.1.2 Greedy Gift Givers(gift1)
这道题大意就是几个人互送礼物,让你求每个人的盈利. 原题给的样例数据: 5(人的个数.) =========(下面是人名,输出按照这顺序)davelauraowenvickamr ========== ...
- USACO Section 1.3 Ski Course Design 解题报告
题目 题目描述 有N座山,每座山都有一个高度,现在由于农夫想避税,所以想把这些山的高度进行一些改变,使得最高的山与最低的山之间的高度差不超过17.每座山最多只能改变一次高度,每次改变高度都会产生一定的 ...
- USACO Section 1.2 Name That Number 解题报告
题目 题目描述 在一个农场里面,每一头牛都有一个数字编号,但是现在这些牛不喜欢这种编号,它们想把这些数字编号转化成为可以接受的字母的形式.数字与字母的转换表如下: 2: A,B,C 5: J,K,L ...
- USACO Section 1.1 Friday the Thirteenth 解题报告
题目 题目描述 黑色星期五是否真的是一件不同寻常的事情?按理来说每个月的13号可能是星期一,或者是星期二...或者是星期天,但是黑色星期五的存在让我们不禁开始猜想,难道每个月的13号刚好是星期五的频率 ...
- USACO Section 1.4 Mother's Milk 解题报告
题目 题目描述 有三个牛奶桶,三个桶的容积分别是A,B,C,最小为1,最大为20.刚开始只有第三个桶里面装满了牛奶,其余两个桶都是空的.我们现在可以将第三个桶中的牛奶往其他两个桶里面倒一些牛奶,然后还 ...
- USACO Training Section 1.1 贪婪的送礼者Greedy Gift Givers
P1201 [USACO1.1]贪婪的送礼者Greedy Gift Givers 题目描述 对于一群(NP个)要互送礼物的朋友,GY要确定每个人送出的钱比收到的多多少.在这一个问题中,每个人都准备了一 ...
随机推荐
- VVDocumenter安装过程的一些问题
vvdocument是瞄神写的一个插件 作用的话大家都知道 这里就不多说了 插件下载地址:https://github.com/onevcat/VVDocumenter-Xcode 1.下载后解压 编 ...
- 事件委托小demo(jq版)
<style type="text/css"> * { margin:; padding:; } .box1 { width: 200px; height: 60px; ...
- 689C - Mike and Chocolate Thieves 二分
题目大意:有四个小偷,第一个小偷偷a个巧克力,后面几个小偷依次偷a*k,a*k*k,a*k*k*k个巧克力,现在知道小偷有n中偷法,求在这n种偷法中偷得最多的小偷的所偷的最小值. 题目思路:二分查找偷 ...
- jquery 功能强大的下拉菜单
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3c.org ...
- 编译使用tinyxml
环境: win7 32位旗舰版,VS2010,tinyxml_2_6_2版本 1.下载tinyxml,并解压到tinyxml文件夹下 2.生成动态链接库 原生的Tinyxml只支持静态库(没有在.h文 ...
- 基础-session,cookie,jsp,EL,JSTL
会话可以简单理解为:用户打开一个浏览器,点击多个超链接,访问服务器多个web资源,然后关闭浏览器,整个过程称之为一个会话. kookie是在服务器端创建的,返回给浏览器,在浏览器的目录中保存了,下一次 ...
- frameset导航框架
1.制作导航框架(注意"name='mainframe'") <html> <frameset cols="25%,75%"> < ...
- FZU 2140 Forever 0.5(将圆离散化)
主要就是将圆离散化,剩下的都好办 #include<iostream> #include<cstdio> #include<cstring> #include< ...
- java synchronized内置锁的可重入性和分析总结
最近在读<<Java并发编程实践>>,在第二章中线程安全中降到线程锁的重进入(Reentrancy) 当一个线程请求其它的线程已经占有的锁时,请求线程将被阻塞.然而内部锁是可重 ...
- Oracle Sql优化之范围处理
1.表中字段自关联与分析函数的性能比较,自关联需要扫描表两次,分析函数扫描一次即可 ----自关联 select v1.proj_id,v1.proj_start,v1.proj_end from v ...