hdu1325 Is It A Tree?并检查集合

职务地址

试想一下,在词和话题hdu1272是一样的。

可是hdu1272的博文中我也说了。数据比較水,所以我用非并查集的方法就AC了。

可是这题的数据没那么水,要用到并查集来解。这题的盲点和重点有这么几个:

  • 输入不是以-1 -1结束,而是以两个负数结束
  • 须要用并查集来推断是不是仅仅有一个“根”
  • 须要推断全部节点的入度是否大于1
  • 本题输入的格式。也要注意一下。
能够先忽略图中的方向的。由于假设有环的话,就变成图了。若严格的依照用parent数组来保存上一级父节点的方法,会有冲突。


比方这个图中,8的父节点有两个那么parent[8]应该怎样存储呢??所以我们能够把它看成无向图,无所谓父子。仅仅需把关系集合merge合并就好了。这样的有多个父节点的情况,就使用一个记录入度的数组来标记就好了。

#include<iostream>
using namespace std;
const int MAX=1000;
int rudu[MAX+10];//入度
int parent[MAX+10];
int root[MAX+10];//保存是否为根
bool flag = true;
int r=0;
int getParent(int a)
{
	int k=parent[a];
	if(parent[a]!=a)
	{
		parent[a]=getParent(parent[a]);
	}
	return parent[a];
}
void merge(int a,int b)
{
	int p1=getParent(a);
	int p2=getParent(b);
	if(p1==p2)
		return;
	parent[p1]=p2;
	root[p2]=1;
	root[p1]=0;
}
void init()
{
	r=0;
	flag = true;
	for(int i=0;i<=MAX;i++)
	{
		parent[i]=i;
		rudu[i]=0;
		root[i]=0;
	}
}

void main()
{
	int a,b,c=1;

	init();
	while(cin>>a>>b,a>=0&&b>=0)
	{
		if(!a&&!b)
		{
			for(int i=0;i<=MAX;i++)
			{
				if(rudu[i]>1)
					flag = false;
				if(root[i])
					r++;
			}
			if(r>1)
				flag=false;
			if(flag)
				cout<<"Case "<<c++<<" is a tree."<<endl;
			else
				cout<<"Case "<<c++<<" is not a tree."<<endl;
			init();
			continue;
		}
		if(getParent(a)!=getParent(b))
			merge(a,b);
		rudu[b]++;
	}
}


版权声明:本文博客原创文章,博客,未经同意,不得转载。

posted @ 2015-08-19 20:44  lcchuguo  阅读(475)  评论(0编辑  收藏  举报