天梯赛 L2-007. (并查集) 家庭房产
题目描述
给定每个人的家庭成员和其自己名下的房产,请你统计出每个家庭的人口数、人均房产面积及房产套数。
输入格式:
输入第一行给出一个正整数N(<=1000),随后N行,每行按下列格式给出一个人的房产:
编号 父 母 k 孩子1 ... 孩子k 房产套数 总面积
其中 编号 是每个人独有的一个4位数的编号;父 和 母 分别是该编号对应的这个人的父母的编号(如果已经过世,则显示-1);k(0<=k<=5)是该人的子女的个数;孩子i是其子女的编号。
输出格式:
首先在第一行输出家庭个数(所有有亲属关系的人都属于同一个家庭)。随后按下列格式输出每个家庭的信息:
家庭成员的最小编号 家庭人口数 人均房产套数 人均房产面积
其中人均值要求保留小数点后3位。家庭信息首先按人均面积降序输出,若有并列,则按成员编号的升序输出。
输入样例:
10
6666 5551 5552 1 7777 1 100
1234 5678 9012 1 0002 2 300
8888 -1 -1 0 1 1000
2468 0001 0004 1 2222 1 500
7777 6666 -1 0 2 300
3721 -1 -1 1 2333 2 150
9012 -1 -1 3 1236 1235 1234 1 100
1235 5678 9012 0 1 50
2222 1236 2468 2 6661 6662 1 300
2333 -1 3721 3 6661 6662 6663 1 100
输出样例:
3
8888 1 1.000 1000.000
0001 15 0.600 100.000
5551 4 0.750 100.000
分析:
可以发现家族成员间关系的处理应用并查集就可以解决,但是关键在于对于家族中的房产信息处理。
我们用一个人的编号来唯一的表示一个人,用编号作为下标来记录这个人有没有出现过。对于所有的出现过的人的编号,找到他所在的并查集的根节点(也就是它本身的父节点),令其父节点所在家庭的人数加一,并且在此时计算家中的人均房产数和人均房产面积。最终排序后输出所需要的东西即可。
代码:
#include<stdio.h>
#include<iostream>
#include<algorithm>
using namespace std;
struct Person
{
int id;//一个人的id
int h_num;//拥有的房产数
int h_s;//拥有的房产面积
} person[10009];
struct Family
{
int id;//家中最小的成员id
int f_num;//成员个数
int h_num;//家庭房产数
int h_s;//家庭房产面积
double avg_h;//平均房产数
double avg_s;//平均房产面积
} family[10009];
bool cmp(Family a,Family b)
{
if(a.avg_s==b.avg_s)
return a.id<b.id;
return a.avg_s>b.avg_s;
}
int pre[10009];//父节点
int vis[10009];//标记这个编号的人有没有出现过
int find_pre(int x)
{
while(x!=pre[x])
x=pre[x];
return x;
}
void unin(int x,int y)
{
int px=find_pre(x);
int py=find_pre(y);
if(px!=py)
pre[px]=py;
}
void init()
{
for(int i=0; i<10000; i++)
{
pre[i]=i;
family[i].id=0;
family[i].f_num=0;
family[i].h_num=0;
family[i].h_s=0;
family[i].avg_h=0;
family[i].avg_s=0;
vis[i]=0;
}
}
int main()
{
init();
int n,a,b,c,k,child;
scanf("%d",&n);
for(int i=0; i<n; i++)
{
scanf("%d%d%d",&a,&b,&c);//本身和双亲的编号
vis[a]=1;
if(b!=-1)//父亲存在
{
vis[b]=1;
unin(a,b);
}
if(c!=-1)//母亲存在
{
vis[c]=1;
unin(a,c);
}
scanf("%d",&k);//孩子的个数,并且要建立联系
for(int j=0; j<k; j++)
{
scanf("%d",&child);
vis[child]=1;
unin(a,child);
}
person[i].id=a;
scanf("%d%d",&person[i].h_num,&person[i].h_s);
}
for(int i=0; i<n; i++)
{
int Id=find_pre(person[i].id);//家族的编号
family[Id].h_num+=person[i].h_num;
family[Id].h_s+=person[i].h_s;
}
for(int i=0; i<10000; i++)
{
if(vis[i]==1)//该人存在
{
pre[i]=find_pre(i);
if(family[pre[i]].f_num==0)//这里保证了此时的家庭的id就是家中的所有人的最小编号
{
family[pre[i]].id=i;
}
family[pre[i]].f_num++;
family[pre[i]].avg_h=family[pre[i]].h_num*1.0/family[pre[i]].f_num;
family[pre[i]].avg_s=family[pre[i]].h_s*1.0/family[pre[i]].f_num;
}
}
sort(family,family+10000,cmp);
int ans=0;
for(int i=0;i<n;i++)
{
if(family[i].f_num!=0)
ans++;
else
break;
}
printf("%d\n",ans);
for(int i=0;i<ans;i++)
printf("%04d %d %.3lf %.3lf\n",family[i].id,family[i].f_num,family[i].avg_h,family[i].avg_s);
return 0;
}
天梯赛 L2-007. (并查集) 家庭房产的更多相关文章
- 天梯赛 L2-024. (并查集) 部落
题目链接 题目描述 在一个社区里,每个人都有自己的小圈子,还可能同时属于很多不同的朋友圈.我们认为朋友的朋友都算在一个部落里,于是要请你统计一下,在一个给定社区中,到底有多少个互不相交的部落?并且检查 ...
- 天梯赛 L2-013. (并查集) 红色警报
题目链接 题目描述 战争中保持各个城市间的连通性非常重要.本题要求你编写一个报警程序,当失去一个城市导致国家被分裂为多个无法连通的区域时,就发出红色警报.注意:若该国本来就不完全连通,是分裂的k个区域 ...
- PTA|团体程序设计天梯赛-练习题目题解锦集(C/C++)(持续更新中……)
PTA|团体程序设计天梯赛-练习题目题解锦集(持续更新中) 实现语言:C/C++: 欢迎各位看官交流讨论.指导题解错误:或者分享更快的方法!! 题目链接:https://pintia.cn/ ...
- Travel(HDU 5441 2015长春区域赛 带权并查集)
Travel Time Limit: 1500/1000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others)Total Su ...
- PTA天梯赛L2
L2-001 紧急救援 题意:就是给你一张n<500的图:让你求最短路径,最短路条数,以及路径: 做法,先用dijkstra求最短路,然后dfs找最短路条数,以及点权的最大值: 一般dfs不就可 ...
- 蓝桥杯模拟赛 引爆炸弹-并查集+DFS
引爆炸弹 在一个 n×m的方格地图上,某些方格上放置着炸弹.手动引爆一个炸弹以后,炸弹会把炸弹所在的行和列上的所有炸弹引爆,被引爆的炸弹又能引爆其他炸弹,这样连锁下去. 现在为了引爆地图上的所有炸弹, ...
- 2019湘潭校赛 G(并查集)
要点 题目传送 题目本质是每个点必属于两个集合中的一个,伴随的性质是:如果一个人说别人true,则他们一定属于同一阵营:如果说别人fake,一定不属于同一阵营. 每个点拆为\(i\)和\(i + n\ ...
- 团体程序设计天梯赛-练习集L2-007. 家庭房产
L2-007. 家庭房产 时间限制 400 ms 内存限制 65536 kB 代码长度限制 8000 B 判题程序 Standard 作者 陈越 给定每个人的家庭成员和其自己名下的房产,请你统计出每个 ...
- 天梯赛决赛 L2-1.红色警报 并查集
L2-013. 红色警报 时间限制 400 ms 内存限制 65536 kB 代码长度限制 8000 B 判题程序 Standard 作者 陈越 战争中保持各个城市间的连通性非常重要.本题要求你编写一 ...
随机推荐
- dbgrid控件如何能在左边显示行号?
procedure TMSWageEdit.aqyMSWageEditCalcFields(DataSet: TDataSet);begin inherited; with DataSet do ...
- POJ1637_Sightseeing tour
给一个联通图,有的是单向边,有的是双向边,问是否存在欧拉回路. 乍一看毫无思路,可以这样来搞,对于每条无向边,我们随便指定一个方向,看看是否能够做到所有点的度数之和为偶数. 接下来,对于我们指定的边, ...
- AtCoder Regular Contest 081
C - Make a Rectangle 从大到小贪心即可. # include <bits/stdc++.h> using namespace std; map<int,int&g ...
- BZOJ5286 HNOI/AHOI2018转盘(分块/线段树)
显然最优走法是先一直停在初始位置然后一次性走完一圈.将序列倍长后,相当于找一个长度为n的区间[l,l+n),使其中ti+l+n-1-i的最大值最小.容易发现ti-i>ti+n-(i+n),所以也 ...
- BZOJ3240 NOI2013矩阵游戏(数论)
必修五题. // luogu-judger-enable-o2 #include<iostream> #include<cstdio> #include<cmath> ...
- NOIP赛前集训营-提高组(第一场)#B 数数字
题目描述 小N对于数字的大小一直都有两种看法.第一种看法是,使用字典序的大小(也就是我们常用的判断数字大小的方法,假如比较的数字长度不同,则在较短一个前面补齐前导0,再比较字典序),比如43<3 ...
- 左连接,右连接和等值连接(left join,right join和inner join)
left join(左联接) 返回包括左表中的所有记录和右表中联结字段相等的记录 right join(右联接) 返回包括右表中的所有记录和左表中联结字段相等的记录inner join(等值连接) 只 ...
- selenium+python定位元素的方法及使用
selenium的八种定位方法 By.ID 通过id定位 By.CLASS_NAME 通过className定位 By.CSS_SELECTOR 通过CSS定位 By.LINK_TEXT 通过link ...
- python爬虫headers设置后无效解决方案
此次遇到的是一个函数使用不熟练造成的问题,但有了分析工具后可以很快定位到问题(此处推荐一个非常棒的抓包工具fiddler) 正文如下: 在爬取某个app数据时(app上的数据都是由http请求的),用 ...
- 一类dp的网格模型
关于形如\(f_{i,j} = \sum_{t=1}^{|w|}\sum_{k=1}^{|v|}f_{i+w_t,j+v_k}\),其中\(w_t,v_k\)为一个定值的\(dp\)转移. 可以考虑放 ...