题目地址:http://ac.jobdu.com/problem.php?pid=1446

题目描述:

One way that the police finds the head of a gang is to check people's phone calls. If there is a phone call between A and B, we say that A and B is related. The weight of a relation is defined to be the total time length of
all the phone calls made between the two persons. A "Gang" is a cluster of more than 2 persons who are related to each other with total relation weight being greater than a given threthold K. In each gang, the one with maximum total weight is the head. Now
given a list of phone calls, you are supposed to find the gangs and the heads.

输入:

For each case, the first line contains two positive numbers N and K (both less than or equal to 1000), the number of phone calls and the weight threthold, respectively. Then N lines follow, each in the following format:

Name1 Name2 Time

where Name1 and Name2 are the names of people at the two ends of the call, and Time is the length of the call. A name is a string of three capital letters chosen from A-Z. A time length is a positive integer which is no more
than 1000 minutes.

输出:

For each test case, first print in a line the total number of gangs. Then for each gang, print in a line the name of the head and the total number of the members. It is guaranteed that the head is unique for each gang. The
output must be sorted according to the alphabetical order of the names of the heads.

样例输入:
8 59
AAA BBB 10
BBB AAA 20
AAA CCC 40
DDD EEE 5
EEE DDD 70
FFF GGG 30
GGG HHH 20
HHH FFF 10
8 70
AAA BBB 10
BBB AAA 20
AAA CCC 40
DDD EEE 5
EEE DDD 70
FFF GGG 30
GGG HHH 20
HHH FFF 10
样例输出:
2
AAA 3
GGG 3
0
来源:
2012年浙江大学计算机及软件工程研究生机试真题

#include <stdio.h>
#include <string.h>
#include <stdlib.h> typedef struct node{
char name[4]; //人名
int time; //通话时间
int father; //父节点
int rank; //秩
}Node; typedef struct head{
char name[4]; //头领姓名
int num; //人数
}Head; Node people[1010];
Head gang[1010]; //头领
int num; //总人数 int FindSet (int x){//查询节点属于那个集合
if (people[x].father != x){
people[x].father = FindSet (people[x].father);
}
return people[x].father;
} void Union (int x, int y){//合并两节点
x = FindSet (x);
y = FindSet (y);
if (x == y)
return;
if (people[x].rank > people[y].rank){
people[y].father = x;
people[x].rank += people[y].rank;
}
else{
if (people[x].rank == people[y].rank){
++people[y].rank;
}
people[x].father = y;
}
} int compare1 (const void * p, const void * q){
Node * p1 = (Node *)p;
Node * q1 = (Node *)q;
return p1->father - q1->father;
} int compare2 (const void * p, const void * q){
Head * p1 = (Head *)p;
Head * q1 = (Head *)q;
return strcmp(p1->name, q1->name);
} int main(void){
int N;
int K;
char name1[4], name2[4];
int time;
int i, j, k;
int pre;
int sum; //总的通话时间 while (scanf ("%d%d", &N, &K) != EOF){
num = 0;
while (N-- != 0){
scanf ("%s%s%d", name1, name2, &time);
///////////////////////////////////////////////////////////////////
//查询输入姓名是否已在表中,如果不在,添加进表;否则,更新节点信息
for (i=0; i<num; ++i){
if (strcmp (people[i].name, name1) == 0){
break;
}
}
if (i == num){
strcpy (people[i].name, name1);
people[i].time = time;
people[i].father = i;
people[i].rank = 0;
++num;
}
else{
people[i].time += time;
}
///////////////////////////////////////////////
for (j=0; j<num; ++j){
if (strcmp (people[j].name, name2) == 0){
break;
}
}
if (j == num){
strcpy (people[j].name, name2);
people[j].time = time;
people[j].father = j;
people[j].rank = 0;
++num;
}
else{
people[j].time += time;
}
///////////////////////////////////////////////////////////////////
Union (i, j);//合并节点
}
//最后一次更新各个集合
for (i=0; i<num; ++i){
FindSet (i);
}
//按父节点大小排序
qsort (people, num, sizeof(Node), compare1);
i = j = k = 0;
//找到各个集合的头领及人数
while (i < num){
strcpy(gang[k].name, people[i].name);
pre = i;
++j;
sum = people[i].time;
while ((j < num) && (people[j].father == people[i].father)){
if (people[j].time > people[pre].time){
strcpy (gang[k].name, people[j].name);
pre = j;
}
sum += people[j].time;
++j;
}
if (j - i > 2 && (sum >> 1) > K){//每个时间在每个人处都加了一遍(即各个时间被加了两遍),所以sum >> 1
gang[k].num = j - i;
++k;
}
i = j;
}
printf ("%d\n", k);
qsort (gang, k, sizeof(Head), compare2);
for (i=0; i<k; ++i){
printf ("%s %d\n", gang[i].name, gang[i].num);
}
} return 0;
}

参考资料:并查集 -- 学习详解

九度OJ 1446 Head of a Gang -- 并查集的更多相关文章

  1. 九度OJ小结

    1. 高精度问题 可参考题目 题目1137:浮点数加法   http://ac.jobdu.com/problem.php?pid=1137 对于高精度问题可以考虑使用结构体.上述为浮点数加法,因此该 ...

  2. 九度oj 题目1087:约数的个数

    题目链接:http://ac.jobdu.com/problem.php?pid=1087 题目描述: 输入n个整数,依次输出每个数的约数的个数 输入: 输入的第一行为N,即数组的个数(N<=1 ...

  3. 九度OJ 1502 最大值最小化(JAVA)

    题目1502:最大值最小化(二分答案) 九度OJ Java import java.util.Scanner; public class Main { public static int max(in ...

  4. 九度OJ,题目1089:数字反转

    题目描述: 12翻一下是21,34翻一下是43,12+34是46,46翻一下是64,现在又任意两个正整数,问他们两个数反转的和是否等于两个数的和的反转. 输入: 第一行一个正整数表示测试数据的个数n. ...

  5. 九度OJ 1500 出操队形 -- 动态规划(最长上升子序列)

    题目地址:http://ac.jobdu.com/problem.php?pid=1500 题目描述: 在读高中的时候,每天早上学校都要组织全校的师生进行跑步来锻炼身体,每当出操令吹响时,大家就开始往 ...

  6. 九度OJ 1531 货币面值(网易游戏2013年校园招聘笔试题) -- 动态规划

    题目地址:http://ac.jobdu.com/problem.php?pid=1531 题目描述: 小虎是游戏中的一个国王,在他管理的国家中发行了很多不同面额的纸币,用这些纸币进行任意的组合可以在 ...

  7. 九度OJ 1024 畅通工程 -- 并查集、贪心算法(最小生成树)

    题目地址:http://ac.jobdu.com/problem.php?pid=1024 题目描述:     省政府"畅通工程"的目标是使全省任何两个村庄间都可以实现公路交通(但 ...

  8. 九度OJ 1371 最小的K个数 -- 堆排序

    题目地址:http://ac.jobdu.com/problem.php?pid=1371 题目描述: 输入n个整数,找出其中最小的K个数.例如输入4,5,1,6,2,7,3,8这8个数字,则最小的4 ...

  9. 九度OJ 题目1384:二维数组中的查找

    /********************************* * 日期:2013-10-11 * 作者:SJF0115 * 题号: 九度OJ 题目1384:二维数组中的查找 * 来源:http ...

随机推荐

  1. linux安装oracle

    目 录 一.硬件要求二.软件三.系统安装注意四.安装Oracle前的系统准备工作五.安装Oracle,并进行相关设置六.升级Oracle到patchset 10.2.0.4七.使用rlwrap调用sq ...

  2. 教你50招提升ASP.NET性能(十):减少通过网络发送的数据

    (16)Reduce the data sent across the network 招数16: 减少通过网络发送的数据 Reducing the amount of data sent acros ...

  3. IOS7 隐藏状态栏

    - (UIStatusBarStyle)preferredStatusBarStyle { return UIStatusBarStyleLightContent; } // - (BOOL)pref ...

  4. VS项目如何运用svn的忽略列表

    在实际的项目开发中,有些文件(比如bin,obj下的文件)是不需要放在svn里面的,因为每次都会重新生成. 该如何排除这些文件那? 我试着在svn server上删除了这些文件夹,但是在文件夹上还是显 ...

  5. [Heroku] How to pull, push changes

    1. First you need to login heroku: heroku login 2. Then you need to download the code: heroku git:cl ...

  6. iOS开发——OC篇&协议篇/NSCoder/NSCoding/NSCoping

    协议篇/NSCoder/NSCoding/NSCoping 协议声明类需要实现的的方法,为不同的类提供公用方法,一个类可以有多个协议,但只能有一个父类,即单继承.它类似java中的接口. 正式协议(f ...

  7. android学习日记13--数据存储之SQLite

    2.SQLite 开源轻量级数据库,支持92-SQL标准,主要用于嵌入式系统,只占几百K系统资源此外,SQLite 不支持一些标准的 SQL 功能,特别是外键约束(FOREIGN KEY constr ...

  8. (function(){}).call(window) 严格模式匿名函数的this指向undefined

    上次在群里,看到有人发出 (function(){}).call(window) 这么一段代码,问这有什么意义,匿名函数中的this不是始终都指向window的么,为什么还要call,我当时也很疑惑. ...

  9. 学习笔记之Shell脚本的输出重定向

    shell http://baike.baidu.com/link?url=qN3THt5ZJhQtwRJJkakWdz5-vZp4V9H3OmNP97XNhaoL-dqU-6rrFrYfHXmYv6 ...

  10. cocos2dx 公告效果

    第一种方法: http://blog.csdn.net/jackystudio/article/details/12991977 [玩转cocos2d-x之十六]滚动字幕和公告 第二种方法: http ...