1362:家庭问题(family)

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 10
  1 2
  1 3
  3 4
  2 5
  3 2
  3 6
  7 8
  7 9
  8 10
  7 2
 */

 

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值