并查集的经典题目。

并查集。经典题目是HDU1232通畅工程。

题目描述:

某省调查城镇交通状况,得到现有城镇道路统计表,表中列出了每条道路直接连通的城镇。省政府“畅通工程”的目标是使全省任何两个城镇间都可以实现交通(但不一定有直接的道路相连,只要互相间接通过道路可达即可)。问最少还需要建设多少条道路?

Output:

对每个测试用例,在1行里输出最少还需要建设的道路数目。

Sample Input:

测试输入包含若干测试用例。每个测试用例的第1行给出两个正整数,分别是城镇数目N ( < 1000 )和道路数目M;随后的M行对应M条道路,每行给出一对正整数,分别是该条道路直接连通的两个城镇的编号。为简单起见,城镇从1到N编号。

注意:两个城市之间可以有多条道路相通,也就是说

3 3

1 2

1 2

2 1

这种输入也是合法的

当N为0时,输入结束,该用例不被处理。

Sample Output:

4 2

1 3

4 3

3 3

1 2

1 3

2 3

5 2

1 2

3 5

999 0

0

思路

相互连通的城市组成城市群。城市群有一个代表。不断输入道路意味着城市之间的互联,如果这两个城市原来属于不同城市群则发生城市群合并时间。问询某个城市的城市群id,依赖于其“父节点”,父节点再从它的父节点询问,直到Boss节点。

find过程:查找城市群Boss id的过程。存在路径优化。

union过程:合并原有的两个城市群。利用rank合并,避免最坏情况线性N级树结构导致find低效。

一个union的case:

       (3)
A----------C
/|\ /|\
| |
|(1) |(2)
| |
B D
/|\
|(4)
|
E

代码

翻出7年前的AC代码,发现没有路径优化、各种全局变量、

//hdu1213
//find_union_set
#include <iostream>
#include <stdio.h>
using namespace std;
int p[1001];
int find(int x)
{
return p[x]==x?x:p[x]=find(p[x]);
}
void Union(int a,int b)
{
p[find(a)]=find(b);
}
int main()
{
// freopen("Chris.txt","r",stdin);
int n,m;
while(scanf("%d",&n)!=EOF)
{
if(n==0)break;
scanf("%d",&m);
int i;
for(i=0;i<=n;i++)
p[i]=i;
for(i=0;i<m;i++)
{
int a,b;
scanf("%d%d",&a,&b);
if(find(a)!=find(b))
{
Union(a,b);
n--;
}
}
printf("%d\n",n-1);
}
return 0;
}

考虑去掉全局变量,添加路径压缩、利用rank优化union过程、封装为结构体、添加注释,代码如下:

#include <stdio.h>
#include <string.h> #define MAXN 1005 //FSU: Find-Union-Set algorithm node struct
typedef struct FSU_Node FSU_Node;
struct FSU_Node{
int p; // parent id
int rank; // rank is defined as number of childern nodes
}; //find node (whose id is x)'s belonging group's root node id
// @param x: node id
// @param nodes: all the nodes (contain several node groups)
int fus_find(int x, FSU_Node* nodes) {
if (nodes[x].p == x) return x;
nodes[x].p = fus_find(nodes[x].p, nodes); // squeeze the finding path
return nodes[x].p;
} //merge two node groups
// @param a: a node from one node group
// @param b: a node from another node group
void fus_union(int a, int b, FSU_Node* nodes) {
int root_a = fus_find(a, nodes);
int root_b = fus_find(b, nodes);
if (root_a == root_b) return; // merge node groups according to rank
// which avoids the worst case: all nodes in one line (so many level of nodes)
if (nodes[root_a].rank > nodes[root_b].rank) {
nodes[root_b].p = root_a;
}
else {
if (nodes[root_a].rank == nodes[root_b].rank) {
nodes[root_b].rank++;
}
nodes[root_a].p = root_b;
}
} #define DEBUG
int main() {
#ifdef DEBUG
freopen("F:/zhangzhuo/debug/OJ/HDU-1232.txt", "r", stdin);
#endif FSU_Node nodes[MAXN]; int n, m, i, j, k;
while (scanf("%d", &n) && n) {
for (i = 0; i <= n; i++) {
nodes[i].p = i;
nodes[i].rank = 1;
} scanf("%d", &m);
int cityId1, cityId2; for (i = 0; i < m; i++) {
scanf("%d %d", &cityId1, &cityId2);
if (fus_find(cityId1, nodes) != fus_find(cityId2, nodes)) {
fus_union(cityId1, cityId2, nodes);
n--;
}
}
printf("%d\n", n - 1);
} return 0;
}

HDU1213通畅工程-并查集求解的更多相关文章

  1. HDU1232 畅通工程 并查集

    畅通工程 Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submis ...

  2. ACM: 继续畅通工程-并查集-最小生成树-解题报告

    继续畅通工程 Time Limit:1000MS Memory Limit:32768KB 64bit IO Format:%I64d & %I64u Submit Status Descri ...

  3. ACM: 畅通工程-并查集-解题报告

    畅通工程 Time Limit:2000MS Memory Limit:32768KB 64bit IO Format:%I64d & %I64u Description 某省调查城镇交通状况 ...

  4. B - 畅通工程(并查集)

    对并查集理解之后就可以做这种题了,虽说这种题做的不多,这道题做过才这么快搞定,可是还是挺happy滴,加油 Description 某省调查城镇交通状况,得到现有城镇道路统计表,表中列出了每条道路直接 ...

  5. NSOJ 畅通工程(并查集)

    某省调查城镇交通状况,得到现有城镇道路统计表,表中列出了每条道路直接连通的城镇.省政府“畅通工程”的目标是使全省任何两个城镇间都可以实现交通(但不一定有直接的道路相连,只要互相间接通过道路可达即可). ...

  6. hdu 1233 还是畅通工程 并查集or最小生成树

    某省调查乡村交通状况,得到的统计表中列出了任意两村庄间的距离.省政府“畅通工程”的目标是使全省任何两个村庄间都可以实现公路交通(但不一定有直接的公路相连,只要能间接通过公路可达即可),并要求铺设的公路 ...

  7. hdu1232 畅通工程 并查集的 应用

    畅通工程 Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submi ...

  8. hdu 1863 畅通工程 (并查集+最小生成树)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1863 畅通工程 Time Limit: 1000/1000 MS (Java/Others)    M ...

  9. HDU - 1232 畅通工程-并查集模板

    某省调查城镇交通状况,得到现有城镇道路统计表,表中列出了每条道路直接连通的城镇.省政府“畅通工程”的目标是使全省任何两个城镇间都可以实现交通(但不一定有直接的道路相连,只要互相间接通过道路可达即可). ...

随机推荐

  1. 【SpringBoot】SpringBoot配置与单元测试(二)

    SpringBoot项目创建参考[SpringBoot]SpringBoot快速入门(一) 本文介绍SpringBoot项目的POM文件.配置与单元测试 POM文件 1.SpringBoot的pom文 ...

  2. 转 Oracle 同一个字段的两值进行加减计算

    https://www.cnblogs.com/hjianguo/p/6041617.html 如 病人ID      入院日期                出院日期 00001      2016 ...

  3. JZ落选跟我们有什么关系

    今天中午睡前刷了一下微博,看到JZ派落选了,底下一大堆冷嘲热讽的. 比如,养了一堆白眼狼,给了XG一堆利好政策,却这样FZ. 这种心态像极了多子女家庭的生活. 多子女家庭里,总有几个是性格比较乖巧,也 ...

  4. Transaction-Mybatis源码

    github地址:https://github.com/dchack/Mybatis-source-code-learn (欢迎star) TransactionFactory 官方文档: 在 MyB ...

  5. 使用MAT 分析内存泄漏实战

    在分析前可以阅读下这篇文章,很多内容就都知道了https://www.jianshu.com/p/738b4f3bc44b,下面我来分析上一篇文章内存泄漏的为问题 1.上一篇中已经生成了hprf的文件 ...

  6. oracle 导出,导入表

    导出 exp DZQZ/DZQZ@orcl file=D:/DZQZ.dmp log=D:/DZQZ.log 导入 imp DZQZ/DZQZ@orcl file=D:\电子取证\DZQZ.dmp f ...

  7. (转)搭建Elasticsearch和kibana环境

    搭建Elasticsearch和kibana环境 作者:IT云清 原文:https://blog.csdn.net/weixin_39800144/article/details/81162002 1 ...

  8. 使用 Nginx 阻止恶意 IP 访问

    找到具有明显特征的访问记录,比如: /Dec/::: +] "-" "Ouija_x.86/2.0" "-" 也许是某个开源框架的漏洞,执行 ...

  9. haproxy转发真实IP给web

    1.在haproxy.cfg中加入下面参数. option forwardfor               #如果后端服务器需要获得客户端真实ip需要配置的参数,必须要放在listen模块下 2.如 ...

  10. 027 SSM综合练习03--数据后台管理系统--product-list.jsp和main.jsp页面制作

    1.product-list.jsp页面制作 (1)创建一个product-list1.jsp文件,清空,只保留 <%@ page contentType="text/html;cha ...