畅通工程
Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)
4 21 33 34 31 25 21 3 2 3 1 203 5999 0
109982
并查集由一个整数型的数组和两个函数构成。数组pre[]记录了每个点的前导点是什么,函数find是查找,join是合并。
int pre[1000 ];
int find(int x) //查找根节点
{
int r=x;
while ( pre[r ] != r ) //返回根节点 r
r=pre[r ];
int i=x , j ;
while( i != r ) //路径压缩
{
j = pre[ i ]; // 在改变上级之前用临时变量 j 记录下他的值
pre[ i ]= r ; //把上级改为根节点
i=j;
}
return r ;
}
void join(int x,int y) //判断x y是否连通,
//如果已经连通,就不用管了 //如果不连通,就把它们所在的连通分支合并起,
{
int fx=find(x),fy=find(y);
if(fx!=fy)
pre[fx ]=fy;
}
#include<stdio.h>
#include<string.h>
int pre[1010],b[1001];
int find(int x) //查找根节点
{
int r=x;
while(pre[r]!=r) //返回根节点 r
r=pre[r];
int i=x,j;
while(i!=r) //路径压缩
{
j=pre[i];
pre[i]=r;
i=j;
}
return r;
}
void join(int x,int y) //判断x y是否连通,
{ //如果已经连通,就不用管了
//如果不连通,就把它们所在的连通分支合并起,
int fx=find(x);
int fy=find(y);
if(fx!=fy)
pre[fx]=fy;
}
int main()
{
int m,n;
while(~scanf("%d",&n))
{
if(n==0)
break;
scanf("%d",&m);
int t,i,x,y,count=0;
for(i=1; i<=n; i++)
pre[i]=i;
for(i=1;i<=m; i++)
{
scanf("%d %d",&x,&y);
join(x,y);
}
memset(b,0,sizeof(b));
for(i=1; i<=n; i++)
{
b[find(i)]=1; //查找有多少个根节点
count+=b[i]; //计数
}
printf("%d\n",count-1);
}
return 0;
}