较为麻烦的并查集
主要是我的模板是错的检查了好久。。。。
先是输入 把每个家庭连在一起
输出的家庭编号为该家庭所有编号的最小值 在并查集里面完成
第一次 0~n-1遍历储存好 家庭编号 和房子面积和数量
第二次0~N遍历 遍历家庭人数
第三遍 处理人均面积和家庭数量和人均数量
#include<bits/stdc++.h> using namespace std; int f[10000]; int find1(int x) { int j=x; while(j!=f[j]) j=f[j]; int cur=x; if(cur!=j) { int t=f[cur]; f[cur]=j; cur=t; } return j; } void union1(int x,int y) { int x1=find1(x); int y1=find1(y); if(x1<y1)f[y1]=x1; else f[x1]=y1; return ; } struct peo { int id; double n,s; int num; int flag; peo() { id=n=s=flag=num=0; } }s[10001],ans[10001]; int vis[10001]={0}; bool cmp(struct peo a,struct peo b) { if(a.s!=b.s)return a.s>b.s; else return a.id<b.id; } int main() { for(int i=0;i<10000;i++)f[i]=i; int k,q; scanf("%d",&k); int a,b,c; for(int i=0;i<k;i++) { scanf("%d%d%d",&a,&b,&c); s[i].id=a; vis[a]=1; if(b!=-1) {union1(a,b);vis[b]=1;} if(c!=-1) {union1(a,c);vis[c]=1;} scanf("%d",&q); while(q--) { scanf("%d",&b); if(b!=-1) {union1(b,a);vis[b]=1;} } scanf("%lf%lf",&s[i].n,&s[i].s); } for(int i=0;i<k;i++) { int a=find1( s[i].id ); ans[ a ].id=a; ans[ a ].s+=s[i].s; ans[ a ].n+=s[i].n; ans[ a ].flag=1; } for(int i=0;i<10000;i++) { if(vis[i]) { int a=find1( i ); ans[a].num+=1; } } int cnt=0; for(int i=0;i<10000;i++) { if(ans[i].flag) { cnt++; ans[i].s=(double)ans[i].s/ans[i].num; ans[i].n=(double)ans[i].n/ans[i].num; } } sort(ans,ans+9999,cmp); printf("%d\n",cnt); for(int i=0;i<cnt;i++) printf("%04d %d %.3lf %.3lf\n",ans[i].id,ans[i].num,ans[i].n,ans[i].s); }
原文:https://www.cnblogs.com/bxd123/p/10369651.html