HDU-1272-小希的迷宫(并查集)

题目链接

http://acm.hdu.edu.cn/showproblem.php?pid=1272

这个题看了别人的思路,虽然AC了但是我自己都不知道为什么。

解题思路:题目意思是找到判断是不是连通无环的图,首先想到的就是并查集。

              1判断成环的时候,只要判断输入边的两个点。有一个共同的父节点,那么这两个点就成环。

              2判断连通的时候,只要判断根节点数为1即可。

             注意:当输入的这组数据只有 0 0 时,依然是满足条件的,即应输出 "Yes"。

对这个并查集,还不是很理解,不过只要理解了代码还是很容易的。

我的AC代码

#include<iostream>
#include<cstring>
#include<cstdio>
#include<algorithm>
using namespace std;

struct node
{
int x,y;
}s[1000000];

int father[1000000];
int hash[1000000];//用来检测是否成环的

int Find(int x)
{
if(x==father[x]) return x;
father[x]=Find(father[x]);
return father[x];
}

void Union(int x,int y)
{
x=Find(x);
y=Find(y);
hash[x]++;//用来检测是否成环的
if(x!=y)
{
father[x]=y;
}
}

int main(void)
{
int n,i,j,k,l;
while(scanf("%d%d",&s[0].x,&s[0].y)==2)
{
memset(hash,0,sizeof(hash));
if(s[0].x==-1&&s[0].y==-1)
return 0;
else if(s[0].x==0&&s[0].y==0)
printf("Yes\n");
else
{
n=1;
while(scanf("%d%d",&s[n].x,&s[n].y)==2)
{
if(s[n].x==0&&s[n].y==0)
break;
n++;
}
for(i=0;i<n;i++)
{
father[s[i].x]=s[i].x;
father[s[i].y]=s[i].y;
}
for(i=0;i<n;i++)
{
Union(s[i].x,s[i].y);
}
k=0;
for(i=0;i<n;i++)
{
if(s[i].x==father[s[i].x])
k++;
if(s[i].y==father[s[i].y])
k++;
}
l=0;
for(i=0;i<n;i++)
{
if(hash[s[i].x]==2)
l++;
if(hash[s[i].y]==2)
l++;
}
if(k==1&&l==0)
printf("Yes\n");
else
printf("No\n");
}
}
return 0;
}

 

我的代码可能很乱;在网上找了一篇代码

可能更能理解一些,代码更好一些。

  #include<iostream>
 2  using namespace std;
 3  #define MAX 100005
 4  int father[MAX],flag,sign[MAX];
 5  
 6  int FindSet(int x)
 7  {
 8      while(x!=father[x])
 9          x=father[x];
10      return x;
11  }
12  
13  void Union(int x,int y)
14  {
15      x=FindSet(x);
16      y=FindSet(y);
17      if(x!=y)
18          father[x]=y;
19      else flag=0; //同父节点,成环
20  }
21  
22  int main()
23  {
24      int i,a,b;
25      while(cin>>a>>b)
26      {
27          if(a==-1&&b==-1) break;
28          if(a==0&&b==0)
29          { cout<<"Yes"<<endl; continue; }
30          for(i=1;i<MAX;i++) 
31          {
32              father[i]=i;
33              sign[i]=0;
34          }
35          sign[a]=sign[b]=1;
36          flag=1;
37          Union(a,b);
38          while(cin>>a>>b)
39          {
40              if(a==0&&b==0) break;
41              Union(a,b);
42              sign[a]=sign[b]=1;
43          }
44          int k=0;
45          for(i=1;i<MAX;i++)
46          {
47              if(sign[i]&&father[i]==i) //判断根节点k数目
48                  k++; 
49              if(k>1) flag=0;
50          }
51          if(flag) cout<<"Yes"<<endl;
52          else cout<<"No"<<endl;
53      }
54      return 0;
55  }
posted @ 2014-08-27 09:03  立刻行动  阅读(279)  评论(0编辑  收藏  举报