今天数据结构课讲了最小生成树的Kruskal算法和Prim算法,不过都只是概念,可能是怕他们听不懂吧,反正算法实现一概不讲...囧

下午抱着《算法导论》跑去图书馆看Kruskal算法,发现《算法导论》真的是牛XXXX的书啊,看完之后豁然开朗,而且惊讶地发现Kruskal算法居然用到了前两天研究的并查集,爽歪歪了...

Kruskal比较适用于稀疏图,是一种贪心算法:为使生成树上边的权值和最小,则应使生成树中每一条边的权值尽可能地小。

具体做法:找出森林中连接任意两棵树的所有边中,具有最小权值的边,如果将它加入生成树中不产生回路,则它就是生成树中的一条边。这里的关键就是如何判断"将它加入生成树中不产生回路"。

《算法导论》提供的一种方法是采用一种"不相交集合数据结构",也就是并查集了。具体的实现看代码好了,反正核心内容就是如果某两个节点属于同一棵树(Find_Set),那么将它们合并(Union)后一定会形成回路。

编写程序:对于如下一个带权无向图,给出所有边以及权值,用kruskal算法求最小生成树。

输入数据:

11
A B 7
A D 5
B C 8
B D 9
B E 7
C E 5
D E 15
D F 6
E F 8
E G 9
F G 11

输出:

A - D : 5
C - E : 5
D - F : 6
A - B : 7
B - E : 7
E - G : 9
Total:39

代码如下,其实代码可以优化的地方很多,例如当生成树的边数已经等于n-1时即可停止循环...因为不是ACM题,故优化省略不写,只当做算法学习...

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
#include <stdio.h>
#include <stdlib.h>#define MAX 100/* 定义边(x,y),权为w */
typedef struct
{int x, y;int w;
}edge;edge e[MAX];
/* rank[x]表示x的秩 */
int rank[MAX];
/* father[x]表示x的父节点 */
int father[MAX];
int sum;/* 比较函数,按权值(相同则按x坐标)非降序排序 */
int cmp(const void *a, const void *b)
{if ((*(edge *)a).w == (*(edge *)b).w){return (*(edge *)a).x - (*(edge *)b).x;}return (*(edge *)a).w - (*(edge *)b).w;
}/* 初始化集合 */
void Make_Set(int x)
{father[x] = x;rank[x] = 0;
}/* 查找x元素所在的集合,回溯时压缩路径 */
int Find_Set(int x)
{if (x != father[x]){father[x] = Find_Set(father[x]);}return father[x];
}/* 合并x,y所在的集合 */
void Union(int x, int y, int w)
{sum += w;if (x == y) return;/* 将秩较小的树连接到秩较大的树后 */if (rank[x] > rank[y]){father[y] = x;}else{if (rank[x] == rank[y]){rank[y]++;}father[x] = y;}
}/* 主函数 */
int main()
{int i, n;int x, y;char chx, chy;/* 读取边的数目 */scanf("%d", &n);getchar();/* 读取边信息并初始化集合 */for (i = 0; i < n; i++){scanf("%c %c %d", &chx, &chy, &e[i].w);getchar();e[i].x = chx - 'A';e[i].y = chy - 'A';Make_Set(i);}/* 将边排序 */qsort(e, n, sizeof(edge), cmp);sum = 0;for (i = 0; i < n; i++){x = Find_Set(e[i].x);y = Find_Set(e[i].y);if (x != y || (!x && !y)){printf("%c - %c : %d\n", e[i].x + 'A', e[i].y + 'A', e[i].w);Union(x, y, e[i].w);}}printf("Total:%d\n", sum);system("pause");return 0;
}

最小生成树kruskal算法并查集版 C语言实现相关推荐

  1. HDOJ 1863畅通工程(最小生成树kruskal算法并查集实现)

    题目:http://acm.hdu.edu.cn/showproblem.php?pid=1863 最小生成树kruskal算法:http://www.zhuoda.org/irini/78592.h ...

  2. 最小生成树KrusKal算法(并查集)

    洛谷p1111链接 克鲁斯卡尔算法的思路就是由森林变成树的过程,其中最主要的就是贪心和并查集的应用. 我们知道链接n个点需要n-1条边,这就满足的最后生成的是一颗树,而不是一个环.在这n-1条边的选择 ...

  3. 最小生成树Kruskal算法+并查集检查连通

    /* 10 6 1 2 6 1 3 1 1 4 5 2 3 5 2 5 3 3 4 5 3 5 6 3 6 4 4 6 2 5 6 6 */// 本例解决最小生成树问题 // 并查集来加快效率 // ...

  4. UVA10034 Freckles【Kruskal算法+并查集】

    In an episode of the Dick Van Dyke show, little Richie connects the freckles on his Dad's back to fo ...

  5. HDU1233 还是畅通工程【Kruskal算法+并查集】

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

  6. HDU1863 畅通工程【Kruskal算法+并查集】

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

  7. HDU1875 畅通工程再续【Kruskal算法+并查集】

    畅通工程再续 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Sub ...

  8. Kruskal 算法——并查集

    需要注意的地方 :将所有的边权从小到大依次排序,按从小到大依次加入边,每次对加入的边进行如下操作: 1.找出边的两点. 2.判断两点的父亲节点是在同一个集合里. 3.如果不在同一个集合,在将两点所在的 ...

  9. java并查集_一个非常实用而且精妙的算法-并查集(java语言实现)

    在学习数据结构的时候,老师多少会提到并查集,他的应用也是超级广泛.本文首先会通过案例来对并查集有一个介绍.然后给出并查集的java实现. 一.并查集原理 话说在江湖上有很多门派,这些门派相互争夺武林霸 ...

最新文章

  1. 2020 年最具潜力的 44 个顶级开源项目,涵盖 11 类 AI 学习框架、平台(附链接)...
  2. SAP C/4HANA五朵云
  3. 758.6G每秒:阿里云成功防御国内最大规模Memcached DDoS反射攻击
  4. windows编程一日一练(2)
  5. .adobe.ETH.btc.frendi.AUF.AYE.qwex后缀勒索病毒
  6. 新书进展和我的决定。
  7. 配置鸿蒙Windows烧录环境 用Hiburn烧录第一个程序
  8. android 显示清除缓存
  9. spring源码核心:DefaultListableBeanFactory
  10. Android状态栏语言,Android实现3种Notification(状态栏通知)
  11. Xcode6中自动布局autolayout和sizeclass的使用
  12. echart添加点击跳转链接事件多次跳转的问题
  13. 基于STM32CUBEMX的淘晶驰串口屏通讯入门
  14. 十个有趣的“大数据”经典数据挖掘案例
  15. python练习题003:圆的面积
  16. python web微信应用(六) 监测微信撤回的消息
  17. Android 获取经纬度。2018年写
  18. Java编写图像浏览器_浏览器下载图像(JAVA代码)
  19. java基础学习——Swing图形化用户界面编程
  20. 【行人轨迹预测数据集——ETH、UCY】

热门文章

  1. 【字符串】字符串查找 ( 蛮力算法 )
  2. 【Android 应用开发】Android开发技巧--Application, ListView排列,格式化浮点数,string.xml占位符,动态引用图片
  3. 吴裕雄 python 机器学习——数据预处理标准化StandardScaler模型
  4. 2016 Multi-University Training Contest 10 [HDU 5861] Road (线段树:区间覆盖+单点最大小)...
  5. codeblock 恢复默认字体设置
  6. IDEA----破解
  7. R语言学习笔记-机器学习1-3章
  8. 为sort函数指定排序规则时注意的问题以及错误的写法
  9. 插入排序之——希尔排序(c/c++)
  10. android位运算简单讲解