hdu 1325 Is It A Tree? (并查集)

本文提供 HDU 1325 的详细解题思路,介绍如何通过并查集和入度计数判断给定的有向边集合是否构成一棵树。包括关键代码实现与算法流程说明。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1325
题意:可以先参考hdu1272 http://blog.csdn.net/qq_36782366/article/details/75044271
由无向边变为了有向边  判断是否是一颗树
思路:加一个数组 记录入度即可  有一个结点的入度如果大于1 则不成立

#include<iostream>
#include<cstdio>
#include<set>
#include<algorithm>
#include<queue>
#include<cstring>
using  namespace std;
#define maxn 100010
int pre[maxn];
int vis[maxn];
int du[maxn];
int num1,num2;
int find(int x)
{
	int r=x;
	while(r!=pre[r])
		r=pre[r];
	int i=x,j;
	while(r!=i)
	{
		j=pre[i];
		pre[i]=r;
		i=j;
	}
	return r;
}
void join(int x,int y)
{
	int fx=find(x);
	int fy=find(y);
	if(fx!=fy)
	{
		pre[fy]=fx;
	}
}
int main()
{
	int n,m;
	int cas=1;
	while(~scanf("%d%d",&n,&m))
	{
		int num1=0,num2=0;
		for(int i=0;i<maxn;i++)
			pre[i]=i;
		if(n==-1&&m==-1)break;
		if(n==0&&m==0)
		{
			printf("Case %d is a tree.\n",cas++);
			continue;
		}
		join(n,m);
		int flag=0;
		memset(vis,false,sizeof vis);
		memset(du,0,sizeof du);
		if(!vis[n])
		{
			vis[n]=true;
			num1++;
		}
		if(!vis[m])
		{
			vis[m]=true;
			num1++;
		}
		if(!du[m])du[m]=1;
		else flag=1;
		num2++;
		while(~scanf("%d%d",&n,&m))
		{

			if(n==0&&m==0)break;
			if(flag)continue;
			if(!vis[n])
			{
				vis[n]=true;
				num1++;
			}
			if(!vis[m])
			{
				vis[m]=true;
				num1++;
			}
			num2++;
			if(!du[m])du[m]=1;
			else flag=1;
			int fm=find(m);
			int fn=find(n);
			if(fm!=fn)
				pre[fm]=fn;
			else flag=1;
		}
		if(flag||abs(num1-num2)!=1)printf("Case %d is not a tree.\n",cas++);
		else printf("Case %d is a tree.\n",cas++);
	}
	return 0;
}


 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值