hdu 1116 欧拉回路 并查集 一组字符串能否首尾相连成一个字符串

主要是欧拉回路的基础知识,用并查集加工处理

注意欧拉回路和并查集的细节判断

不能粘贴复制,一定要理解之后再敲一遍代码,否则浪费更多的时间

#include <stdio.h>
#include <string.h>

int vis[30],in[30],out[30],father[30];

int findx(int t)
{
	if(father[t]!=t)
		father[t]=findx(father[t]);
	return father[t];
}

void merge(int v,int g)
{
	int x,y;

	x=findx(v);
	y=findx(g);

	if(x!=y)
		father[x]=y;
}

int main()
{
	int n,m,s,e,bcn,i,j;

	int p[30];

	scanf("%d",&n);

	while(n--)
	{
		char a[1001];

		scanf("%d",&m);

		memset(out,0,sizeof(out));
		memset(in,0,sizeof(in));
		memset(vis,0,sizeof(vis));

		for(i=0;i<26;i++)
			father[i]=i;

		while(m--)
		{
			scanf("%s",a);

			s=a[0]-'a';

			e=a[strlen(a)-1]-'a';

			merge(s,e);

			out[s]++;

			in[e]++;

			vis[s]=1;

			vis[e]=1;
		}

		for(i=0;i<26;i++)
			father[i]=findx(i);

		for(bcn=0,i=0;i<26;i++)
		{
			if(father[i]==i&&vis[i])
				bcn++;
		}

		//printf("%d\n",bcn);

		if(bcn>1)
		{
			printf("The door cannot be opened.\n"); 
             continue;
		}

		for(j=0,i=0;i<26;i++)
		{
			if(vis[i]&&out[i]!=in[i])
				p[j++]=i;
		}

		if(j==0)
		{
			printf("Ordering is possible.\n"); 
            continue;
		}

		if(j==2 && ( (out[p[0]]-in[p[0]]==1 && in[p[1]]-out[p[1]]==1 )|| 

(out[p[1]]-in[p[1]]==1 && in[p[0]]-out[p[0]]==1 )) ) 
         {  
             printf("Ordering is possible.\n"); 
             continue; 
         } 


		printf("The door cannot be opened.\n"); 

	}
	return 0;
}

  

posted @ 2012-03-29 17:54  shijiwomen  阅读(307)  评论(0编辑  收藏  举报