1362:家庭问题(family)
时间限制: 1000 ms 内存限制: 65536 KB
提交数: 6732 通过数: 3529
【题目描述】
有n个人,编号为1,2,……n,另外还知道存在K个关系。一个关系的表达为二元组(α,β)形式,表示α,β为同一家庭的成员。
当n,k和k个关系给出之后,求出其中共有多少个家庭、最大的家庭中有多少人?
例如:n=6,k=3,三个关系为(1,2),(1,3),(4,5)
此时,6个人组成三个家庭,即:{1,2,3}为一个家庭,{4,5}为一个家庭,{6}单独为一个家庭,第一个家庭的人数为最多。
【输入】
第一行为n,k二个整数(1≤n≤100)(用空格分隔);
接下来的k行,每行二个整数(用空格分隔)表示关系。
【输出】
二个整数(分别表示家庭个数和最大家庭人数)。
【输入样例】
6 3
1 2
1 3
4 5
【输出样例】
3 3
闲话:这道题说实话我当时学队列时没做出来这道题,后来学了并查集才做出来(说实话我不知道这道题为什么被放在队列的分类里)
并查集模板
join函数:
void join(int p,int q)
{int fp = fin(p),fq = fin(q);if(fp != fq) fa[fq] = fp;
}
fin函数:
int fin(int k)
{if(k == fa[k]) return k;return fa[k] = fin(fa[k]);
}
思路:这道题要求输出家庭个数和最大家庭人数,家庭个数很简单,构造好并查集后直接数数有多少人的老大是他自己就知道了。可最大家庭人数该怎么求呢?我也是这里掉了几次坑。我的做法是先把每个人的最牛boss的编号存储在fa[i]中,再排序,然后判断,如果第i个人的最大boss==第i - 1个人的最大boss则说明出现了一个新家庭,把rs与用来数前一个家庭有几人的变量s取最大值,然后将s初始化为1,否则s++。
坑点:最后还需要
rs = max(rs,s);
!因为如果只有 1个家庭,则数最大家庭人数时根据前面的逻辑,fa[i]会一直等于fa[i - 1],此时就会输出1。不行可以注释掉这一行代码试试下面的样例
10 10
1 2
1 3
3 4
2 5
3 2
3 6
7 8
7 9
8 10
7 2
我前面就是因为没有这一行代码而
代码:
#include <bits/stdc++.h>
using namespace std;
int s = 1,n,m,z,x,y,fa[100001],l,gs,rs;
int fin(int k)
{if(k == fa[k]) return k;return fa[k] = fin(fa[k]);
}
void join(int p,int q)
{int fp = fin(p),fq = fin(q);if(fp != fq) fa[fq] = fp;
}
int main()
{scanf("%d%d",&n,&m);for(int i = 1;i <= n;i++) fa[i] = i;while(m--){scanf("%d%d",&x,&y);join(x,y);}for(int i = 1;i <= n;i++){if(fa[i] == i) gs++;fa[i] = fin(i);}sort(fa + 1,fa + 1 + n);//for(int i = 1;i <= n;i++) cout<<fa[i]<<" ";for(int i = 1;i <= n;i++)if(fa[i] == fa[i - 1]) s++;else{rs = max(rs,s);s = 1;}rs = max(rs,s);printf("%d %d",gs,rs);return 0;
}
/*10 101 21 33 42 53 23 67 87 98 107 2*/