并查集的经典题目。

并查集。经典题目是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. SQL调用另一台服务器的表及存储过程(SQL函数openrowset()的使用以及相关问题处理)

    --查询表select * from openrowset('SQLOLEDB', 'IP'; 'sa'; '密码',数据库名称.dbo.表名称) --查询存储--示例1select * from o ...

  2. React.lazy和React.Suspense异步加载组件

    在React16.6中引入了React.lazy和React.Suspense,这两个组件,可以用来实现异步加载组件. 例如: const johanComponent = React.lazy(() ...

  3. vue aes

    npm install crypto-js import CryptoJS from "crypto-js/crypto-js"; const KEY = CryptoJS.enc ...

  4. @vue/cli 4.1.1安装

    按照安装步骤,先卸载,再安装,最终,查看vue -V 的版本都是3.8.2,也就是说并没有安装成功,于是,考虑用yarn去安装 1,首先清除缓存: yarn cache clean 2,yarn设置淘 ...

  5. but only one is allowed(重复处理跨域请求)

    情景:vue的项目中在本地调试项目,在前端的跨域配置没有问题的情况下,出现这样的报错. 解决方案,参考: https://www.cnblogs.com/zsg88/articles/11576324 ...

  6. SQL Server:时间范围查询重叠

    常常碰到要校验数据范围是否存在重叠冲突的情况,典型的场景是房间预订. 假如房间A已经有9月1日-9月10日的预订记录,当其它客人再来预订时,系统必须判断,不能与这个日期范围产生重叠. 有四种情况会产生 ...

  7. spring中RequestBody注解接收参数时用JSONField转参数名无效问题

    问题: 在springboot项目中使用@RequestBody注解接收post请求中body里的json参数的情况.即: @RequestMapping(value = "/get-use ...

  8. Javascript / Nodejs call 和 apply

    call: 改变了函数运行的作用域,即改变函数里面this的指向apply:同call,apply第二个参数是数组结构 例如: this.name = 'Ab'var obj = {name: 'BB ...

  9. 最常见的Java面试题及答案汇总(三)

    上一篇:最常见的Java面试题及答案汇总(二) 多线程 35. 并行和并发有什么区别? 并行是指两个或者多个事件在同一时刻发生:而并发是指两个或多个事件在同一时间间隔发生. 并行是在不同实体上的多个事 ...

  10. PHP ob_gzhandler的理解

    PHP ob_gzhandler的理解那么对于我们这些没有开启mod_deflate模块的主机来说,就只能采用ob_gzhandler函数来压缩了,它的压缩效果和mod_deflate相比,相差很小, ...