最小生成树 Kruscal经典算法
文章作者:ktyanny 文章来源:ktyanny 转载请注明,谢谢合作。
话说ktyanny昨天逃了一天的课,恶补并查集知识,就是为了写出经典得不得了的Kruscal最小生成树。今天早上9点钟爬起来,继续看了下Kruscal算法,顿然茅塞顿开了,哈哈
1、生成树的概念
连通图G的一个子图如果是一棵包含G的所有顶点的树,则该子图称为G的生成树。
生成树是连通图的极小连通子图。所谓极小是指:若在树中任意增加一条边,则将出现一个回路;若去掉一条边,将会使之变成非连通图。 生成树各边的权值总和称为生成树的权。权最小的生成树称为最小生成树。
2、最小生成树的性质
用哲学的观点来说,每个事物都有自己特有的性质,那么图的最小生成树也是不例外的。按照生成树的定义,n 个顶点的连通网络的生成树有 n 个顶点、n-1 条边。
3、构造最小生成树,要解决以下两个问题:
1.尽可能选取权值小的边,但不能构成回路(也就是环)。
2.选取n-1条恰当的边以连接网的 n个顶点。
4、最小生成树Kruscal算法:
用ktyanny的口头陈述为,Kruscal算法实质是非朴素的贪心策略。初始状态,图的每个节点单独作为一个集合,图的边就绪状态为非降序排列。然后一次遍历图中的所有边(u, v),使用并查集的思想(不懂并查集的点击此处 并查集(不相交集合)
进行学习), find_set(u)如果不等于find_set(v),也就是u和v不在同一个集合中,那么相当于判断了添加了边(u, v)不会照成环。好!接下来的工作就是把需要的信息保存起来(比如边的信息之类的),同时合并u和v(使用union_set(u, v)).当所有的顶点都添加到集合中去了,好!算法完毕。
5、Kruscal算法举例演示
假设部分:图中有9个顶点依次为a、b、c、d、e、f、g、h、i,各条边的权值直接看下面的图吧。
大概的过程看下面的步骤:下面给出的图很囧……(是因为我相机还不够高级,等挣钱了买个漂亮的),不过没关系,主要自己看的明白就好了。
看清楚了,箭头指向的是要处理的边,粗线边表示为被接纳的边了。
好了,在这里还是要夸一下《算法导论》这本书。
6、核心部分的代码
贴上比较好点的代码吧:
const int MAX = 100; /*图的顶点个数最大上限*/
typedef struct
{
int x, y;
int w;
}edge;
edge e[MAX];
/*上面描述为图的数据结构*/
/*下面部分为主要部分,根据你的程序的需要惊醒添加吧*/
/*MST_Kruscal算法的实现过程*/
/* 将边排序 */
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)
{
union_set(x, y, e[i].w);
}
}
7、算法分析
用Kruskal算法构造最小生成树的时间复杂度为O(eloge),与网中边的数目e有关,因此,它适用于求稀疏图的最小生成树。
我们还有一种思想比较复杂点的算法,prim算法,详情可以参考:最小生成树 普莱姆(prim)算法
8、相关练习
HDU 1102 Constructing Roads (Kruscal最小生成树)
HDU 1162 Eddy's picture (Kruscal最小生成树)
HDU 1233 还是畅通工程 (Kruscal 最小生成树)
HDU 1301 Jungle Roads (Kruscal 最小生成树)
HDU 1875 畅通工程再续(Kruscal最小生成树)
posted on 2009-12-10 13:29 Ktyanny Home 阅读(3554) 评论(0) 编辑 收藏 举报