并查集的经典题目。

并查集。经典题目是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. libmodbus学习笔记

    libmodbus功能: a fast and portable Modbus library 库下载地址https://libmodbus.org 使用指南1)包含头文件#include <m ...

  2. kubernetes-通过VMware搭建k8s集群遇到的问题

    VMWare版本:14.13 Centos版本:CentOS-7-x86_64-DVD-1810.iso 遇到的问题:ping不通 报could not resolve host这个错误 解决办法:参 ...

  3. 【NPDP笔记】第三章 新产品流程

      3.1 产品开发,风险与汇报的过程,开发实践和流程提升成功率 管控新产品失败的风险,随着成本增加,风险降低 知识能改改进决策,降低风险,决策框架 识别问题与机会 收集信息 组织记录,组织员工 外部 ...

  4. hive 引入第三方包(不重启)

    Jar放入${HIVE_HOME}/auxlib目录 在${HIVE_HOME}中创建文件夹auxlib,然后将自定义jar文件放入该文件夹中. 此方法添加不需要重启Hive.而且比较便捷. 连接方式 ...

  5. SQL2014做数据库主从镜像备份(也可以用于高可用)备忘(非域控)。

    部份内容参考原始文章链接:https://www.cnblogs.com/stragon/p/5643754.html ,同时比较有参考价值的文章:https://blog.csdn.net/sqls ...

  6. 前端与算法 leetcode 28.实现 strStr()

    # 前端与算法 leetcode 28.实现 strStr() 题目描述 28.移除元素 概要 这道题的意义是实现一个api,不是调api,尽管很多时候api的速度比我们写的快(今天这个我们可以做到和 ...

  7. LeetCode 1021. 删除最外层的括号(Remove Outermost Parentheses)

    1021. 删除最外层的括号 1021. Remove Outermost Parentheses 题目描述 有效括号字符串为空 ("")."(" + A + ...

  8. Django框架之第三篇(路由层)--有名/无名分组、反向解析、路由分发、名称空间、伪静态

    一.Django请求生命周期 二.路由层  urls.py url()方法 第一个参数其实就是一个正则表达式,一旦前面的正则匹配到了内容,就不会再往下继续匹配,而是直接执行对应的视图函数. djang ...

  9. jQuery基础事件处理

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  10. 收藏单词TOEFL备份托福英语

    TOEFL托福词汇串讲(文本) alchemy(chem-化学)n. 炼金术 chemistry 化学 alder 赤杨树 联想:older 老人坐在赤杨树下 sloth 树懒 algae n.海藻 ...