【PAT-并查集-水题】L2-007-家庭房产
L2-007. 家庭房产
给定每个人的家庭成员和其自己名下的房产,请你统计出每个家庭的人口数、人均房产面积及房产套数。
输入格式:
输入第一行给出一个正整数N(<=1000),随后N行,每行按下列格式给出一个人的房产:
编号 父 母 k 孩子1 ... 孩子k 房产套数 总面积
其中 编号 是每个人独有的一个4位数的编号;父 和 母 分别是该编号对应的这个人的父母的编号(如果已经过世,则显示-1);k(0<=k<=5)是该人的子女的个数;孩子i是其子女的编号。
输出格式:
首先在第一行输出家庭个数(所有有亲属关系的人都属于同一个家庭)。随后按下列格式输出每个家庭的信息:
家庭成员的最小编号 家庭人口数 人均房产套数 人均房产面积
其中人均值要求保留小数点后3位。家庭信息首先按人均面积降序输出,若有并列,则按成员编号的升序输出。
做题思路:
1、观察最后两行会发现,2222和2333有相同的孩子——他们的关系没法通过父母的关系网连接到一块;
所以输入孩子也应该连接进关系网中,如果在网中,孩子的父母也指向孩子的根节点;
否则,孩子指向他的该行的父或者母。
2/最后并查集记得进行再次全体进行寻根一次!
3、每次新开一个b【】数组时,记得立即初始化一次!!
4.还出了一个段错误! 您的程序发生段错误,可能是数组越界,堆栈溢出(比如,递归调用层数太多)等情况引起!
后来造了一组无厘头的数据:
3
1222 0 23 0 3 300
0 3 -1 0 1 0
3 0 -1 0 1 0
然后程序就炸了,接着仔细想想——搞了半天我没用上并查集的基本操作add()!!
AC程序:(附注释)
#include<iostream>
#include<stdio.h>
#include<string.h>
#include<math.h>
#include<algorithm>
#include<queue>
#include<set>
#include<vector>
#include<map>
#define maxn 400000
#define inf 0x3f3f3f3f
using namespace std;
#define N 10018
struct node{
int mo,fa;
int k;//从0开始的
int kids[];
int suit;
int area;
}a[N];
struct family{
int root;//记录根的id
int minn_id;
int pnum;//人口总数
int suit;double avesuit;
int area;double avearea;
}b[N];
int tree[N];//建立一棵并查集,tree[i]=x,x表示i的父节点
int vis[N];//记录i是否记录进了人数
int getfa(int x){
if(x==tree[x])
return x;
return tree[x]=getfa(tree[x]);
}
void add(int x,int y){
int fa=getfa(x);
int fb=getfa(y);
if(fa!=fb){
tree[fa]=fb;
}
}
void init(family &x){
x.area=;x.suit=;
x.minn_id=;x.pnum=;
}
int fact1id(int i){
int num=i;
if(a[i].mo!=-)
num=min(a[i].mo,num);
if(a[i].fa!=-)
num=min(a[i].fa,num);
for(int j=;j<=a[i].k;j++){
num=min(a[i].kids[j],num);
}
return num;
}
int fact2(int i){
int num=;
if(vis[i]==){//别把自己也忘了!
num++;vis[i]=;
}
if(a[i].mo!=-&&vis[a[i].mo]==){
num++;vis[a[i].mo]=;
} if(a[i].fa!=-&&vis[a[i].fa]==){
num++;vis[a[i].fa]=;
}
for(int j=;j<=a[i].k;j++){
if(vis[a[i].kids[j] ]==)continue;
num++;
vis[a[i].kids[j] ]=;
}
return num;
}
bool cmp(family x,family y){
if(fabs(x.avearea-y.avearea)<1e-)
return x.minn_id<y.minn_id;
return x.avearea>y.avearea;
}
int main(){
int i,j,n;
scanf("%d",&n);
memset(vis,,sizeof(vis));
for(int i=;i<=;i++){
tree[i]=i;
a[i].area=-;
}
while(n--){
int id;
scanf("%d",&id);
scanf("%d%d",&a[id].fa,&a[id].mo);//确认并查集树上的父子关系
if(a[id].fa!=-){
add(id,a[id].fa);
}
if(a[id].mo!=-){
add(id,a[id].mo);
}
getfa(id); scanf("%d",&a[id].k);
for(int i=;i<=a[id].k;i++){//读入他的k个孩子的ID
scanf("%d",&a[id].kids[i]);
add(id,a[id].kids[i]);
}
scanf("%d%d",&a[id].suit,&a[id].area);
}
for(int i=;i<=;i++)//并查集重新调整一次!
getfa(i); int cnt=;//统计家庭数
set<int>s;//记录根节点
for(int i=;i<=;i++){
if(a[i].area==-){//不存在
continue;
}
int fa=tree[i];
int k;//表示开的family数组的序号
if(s.count(fa)>){
for(int j=;j<=cnt;j++){
if(b[j].root==fa){
k=j;break;
}
}
}else{
s.insert(fa);
++cnt;
k=cnt;
b[cnt].root=fa;//需要添加,别忘记了
init(b[k]);
}
//将a[i]数组的内容转移到b[k]中 b[k].area+=a[i].area;
b[k].minn_id=min(b[k].minn_id,fact1id(i) );
b[k].pnum+=fact2(i);
b[k].suit+=a[i].suit;
}
for(int i=;i<=cnt;i++){
b[i].avearea=b[i].area*1.0/(1.0*b[i].pnum);
b[i].avesuit=b[i].suit*1.0/(1.0*b[i].pnum);
}
sort(b+,b++cnt,cmp); printf("%d\n",cnt);
for(int i=;i<=cnt;i++){
printf("%04d %d %.3lf %.3lf\n",b[i].minn_id,b[i].pnum,b[i].avesuit,b[i].avearea);
} return ;
}
【PAT-并查集-水题】L2-007-家庭房产的更多相关文章
- Brain Network (easy)(并查集水题)
G - Brain Network (easy) Time Limit:2000MS Memory Limit:262144KB 64bit IO Format:%I64d & ...
- poj2524(并查集水题)
题目链接:http://poj.org/problem?id=2524 题目大意:学校共有n个同学,告诉你m对同学信仰同一宗教,问这个学校学生信仰宗教的数目最多为多少. 例: Sample Input ...
- POJ2524并查集水题
Description There are so many different religions in the world today that it is difficult to keep tr ...
- HDU1863(Kruskal+并查集水题)
https://cn.vjudge.net/problem/HDU-1863 省政府“畅通工程”的目标是使全省任何两个村庄间都可以实现公路交通(但不一定有直接的公路相连,只要能间接通过公路可达即可). ...
- PAT题解-1118. Birds in Forest (25)-(并查集模板题)
如题... #include <iostream> #include <cstdio> #include <algorithm> #include <stri ...
- PAT甲级 并查集 相关题_C++题解
并查集 PAT (Advanced Level) Practice 并查集 相关题 <算法笔记> 重点摘要 1034 Head of a Gang (30) 1107 Social Clu ...
- 【HDU1231】How Many Tables(并查集基础题)
什么也不用说,并查集裸题,直接盲敲即可. #include <iostream> #include <cstring> #include <cstdlib> #in ...
- poj1182 食物链(并查集 好题)
https://vjudge.net/problem/POJ-1182 并查集经典题 对于每只动物创建3个元素,x, x+N, x+2*N(分别表示x属于A类,B类和C类). 把两个元素放在一个组代表 ...
- PAT甲题题解-1114. Family Property (25)-(并查集模板题)
题意:给出每个人的家庭成员信息和自己的房产个数与房产总面积,让你统计出每个家庭的人口数.人均房产个数和人均房产面积.第一行输出家庭个数,随后每行输出家庭成员的最小编号.家庭人口数.人均房产个数.人均房 ...
随机推荐
- 2019年Java中高级面试题总结(8)
116.什么时候使用访问者模式? 访问者模式用于解决在类的继承层次上增加操作,但是不直接与之关联.这种模式采用双派发的形式来增加中间层. 117.什么时候使用组合模式? 组合模式使用树结构来展示部分与 ...
- [BJOI2019] 删数 [dp转贪心结论+线段树]
题面 传送门 思路 dp部分 以下称合法序列为原题面中可以删空的序列 这个是我在模拟考场上的思路 一开始我是觉得,这个首先可以写成一个dp的形式:$dp[i][j]$表示用$j$个数字填满了目标序列的 ...
- Mkdown常用语法
这篇笔记停了两天,今天开始整理, 写这篇笔记主要是加强下自己的mkdown语法知识, 也当作练手之作. Mkdown 语法简介 标题 [h1 - h6] # ~ ###### 一般# 作为标题, 只有 ...
- Python23之内置函数filter()和map()
首先我们了解一个概念:迭代 迭代是访问集合元素的⼀种⽅式.迭代器是⼀个可以记住遍历的位置的对象.迭代器对象从集合的第⼀个元素开始访问,直到所有的元素被访问完结束.迭代器只能往前不会后退. 我们已经知道 ...
- Java字符串定义及常用方法
String.StringBuffer和StringBuilder String修饰的是不可变的字符串,而StringBuffer和StringBuilder类的对象是可以被修改的. Stri ...
- P5200 [USACO19JAN]Sleepy Cow Sorting
P5200 [USACO19JAN]Sleepy Cow Sorting 题目描述 Farmer John正在尝试将他的N头奶牛(1≤N≤10^5),方便起见编号为1…N,在她们前往牧草地吃早餐之前排 ...
- Linux基础-04-权限
1. 查看文件的权限 1) 使用ls –l命令查看文件上所设定的权限. -rw-r--r-- 1 root root 605 Mar 18 20:28 .jp1.tar.gz 权限信息 属主 属组 文 ...
- 计算机网络自顶向下方法第4章 网络层:数据平面 (Network layer)
4.1 网络层概述 网络层主要功能为转发(将数据从路由器输入接口转移到合适的输出接口)和路由选择(端到端的路径选择),每台路由器都有一张转发表,用最长前缀匹配规则来转发. 4.1.1 转发和路由选择 ...
- 1181: 零起点学算法88——偶数求和(C语言)
一.题目: 题目来源WUSTOJ 二.源代码: #include<stdio.h> int main() { int n, m, num, sum, i, j, k; while (sca ...
- Java8新特性 - Optional容器类
Optional 类(java.util.Optional) 是一个容器类,代表一个值存在或不存在,原来用null 表示一个值不存在,现在Optional 可以更好的表达这个概念.并且可以避免空指针异 ...