题目描述
题解:
这是一道典型的求连通分支数的问题,可以想到利用并查集。
并查集的构建-详解并查集
首先初始化m*n个元素,并合并题目中相连接的元素,
接着通过查找根元素将相同根元素(属于同一集合)的元素标记:令a[m*n]
数组中根元素下标的位置置为1,。
最后遍历统计数组中置为1的个数即可。
代码如下:
#include<iostream>
using namespace std;
const int maxx=1000010;
int par[maxx];
int ran[maxx];
int a[maxx];
void init(int n)
{
for (int i = 1; i <= n; i++)
{
par[i]=i;
ran[i]=0;
}
}
int find(int x)//路径压缩
{
if (par[x]==x)
{
return x;
}
else
{
return par[x]=find(par[x]);
}
}
int find2(int x)//单纯查找
{
while (par[x]!=x)
{
x=par[x];
}
return x;
}
void unite0(int x,int y)
{
x=find(x);
y=find(y);
if(x!=y) par[x]=y;
}
void unite(int x,int y)
{
x=find(x);
y=find(y);
if (x==y)
{
return ;
}
if (ran[x]<ran[y])
{
par[x]=y;
}
else
{
par[y]=x;
if(ran[x]==ran[y]) ran[x]++;
}
}
bool same(int x,int y)
{
return find(x)==find(y);
}
int main()
{
int m,n;
int x,y;
cin>>m>>n;
int k;
cin>>k;
init(m*n);
for (int i = 1; i <= k; i++)
{
cin>>x>>y;
unite(x,y);
}
int sum=0;
for (int i = 1; i <= m*n; i++)
{
a[find(i)]=1;
}
for (int i = 1; i <= m*n; i++)
{
if(a[i]) sum++;
}
cout<<sum<<endl;
return 0;
}