畅通工程(并查集的基础讲解)

题目连接

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

题目大意

多组样例输入,n座城市,m条路,问最少需要修多少条路能使全部城市相互可达。
本质:求图的连通分支数。

解题思路

思路很明确,就是划分块,相互连通的属于同一块,最后数数多少块输出。
难点在于如何存储块?
出来吧!并查集!
大神讲解,绝了!
我就稍微总结一下:
核心代码

int find(int x){//查找x的掌门
    int r=x;//委托 r 去找掌门
    while(r!=fa[r]){//如果r的上级不是r自己(也就是说找到的大侠他不是掌门)
        r=fa[r];// r 就接着找他的上级,直到找到掌门为止
    }
    return r; //找到了掌门
}

void join(int x,int y){//将x,y连在一起(将x,y归于同一门派)
    int rx=find(x),ry=find(y);//找到x,y的掌门
    if(rx!=ry)//掌门不是同一人
        fa[rx]=ry;//让x的掌门当y的掌门的手下
}

优化代码

int find(int x){
    int r=x;
    while(r!=fa[r]){//找根节点
        r=fa[r];
    }
    int i=x,tmp;
    while(i!=r){//路径压缩
        tmp=fa[i];
        fa[i]=r;
        i=tmp;
    }    
    return r;
}

路径压缩(比较难理解,就算理解了也不是很好写)
本质:让x及x的所有上级的父亲全部变成掌门。
代码中,tmp=fa[i];i=tmp; 实现的是i现在指向刚刚i的上级。fa[i]=r; 将i的上级改为掌门。
图片说明

AC代码

#include<bits/stdc++.h>
using namespace std;
const int N=1e3+5;
int n,m;
int pre[N];
int find(int x){
    int r=x;
    while(pre[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){//连接 
    int rx=find(x),ry=find(y);
    if(rx!=ry)
        pre[rx]=ry;
}

void init(){//初始化
    for(int i=1;i<=n;i++)
        pre[i]=i;
}

int main(){
    while(cin>>n>>m&&n){
        init();
        for(int i=1;i<=m;i++){
            int u,v;
            cin>>u>>v;
            join(u,v);
        }

        int cnt=-1;
        for(int i=1;i<=n;i++)
            if(pre[i]==i) cnt++;
        cout<<cnt<<endl;
    }
} 

代码讲解

注意初始化,将所有点的父亲节点(上级)都置为自己。
输入的同时直接将相连的节点通过join函数连接。
算算有多少掌门,掌门数-1就是连通分支数;而掌门的父节点就是自己,所以找全部节点中父亲节点等于自己的节点就是掌门数。

另一简洁的AC代码

#include<bits/stdc++.h>
using namespace std;
const int N=1010;
int fa[N];
int n,m;
int find(int x){
    return fa[x]==x?x:fa[x]=find(fa[x]);
}
void join(int x,int y){
    int rx=find(x),ry=find(y);
    if(rx!=ry) fa[rx]=ry;
}
void init(){
    for(int i=1;i<=n;i++) fa[i]=i;
}
int main(){
    while(cin>>n&&n){
        init();
        cin>>m;
        while(m--){
            int u,v;
            cin>>u>>v;
            join(u,v);            
        }
        int cnt=0;
        for(int i=1;i<=n;i++)
            if(fa[i]==i) cnt++;
        cout<<cnt-1<<endl;
    }    
} 

总结

其实本题不是很难(如果知道用并查集的话,但是我知道用并查集,最后还是不知道怎么算出连通分支数QWQ)
主要是学习并查集,并查集一般用于有关连通区域的题。
最后的最后,我想问上天一句:我什么时候才能成为大佬啊啊啊!!!

全部评论

相关推荐

2024-11-18 10:36
内蒙古民族大学 Java
白菜小丑呜呜:Radis写错了兄弟
点赞 评论 收藏
分享
评论
1
收藏
分享
牛客网
牛客企业服务