Kruskal算法模拟讲解
Kruskal 算法是一个求最小生成树的算法,即求最小的开销等
算法可以这样,要求得最小生成树,那么n个节点只能包括n-1条边
所以我们应该转换为寻找这最短的n-1条边,因此,可以先对所有的
边进行从小到大排序,每次取出一条边来进行试探,看是否够成环,
如果不构成环,那么肯定是最短的路径了,因为每次都是取最小
的边来试探,最终可以求得最小的生成树代价和。
/*
Filename:kruskal.cpp
Author: xiaobing
E-mail: xiaobingzhang29@gmail.com
Date: 2013-08-31
*/
#include<iostream>
#include<string>
#include<string.h>
#include<algorithm>
#include<cstdlib>
#include<list>
#include<set>
#include<vector>
#define N 100
#define INF 1000000
using namespace std; /*
Kruskal 算法是一个求最小生成树的算法,即求最小的开销等
算法可以这样,要求得最小生成树,那么n个节点只能包括n-1条边
所以我们应该转换为寻找这最短的n-1条边,因此,可以先对所有的
边进行从小到大排序,每次取出一条边来进行试探,看是否够成环,
如果不构成环,那么肯定是最短的路径了,因为每次都是取最小
的边来试探,最终可以求得最小的生成树代价和。
用到的数据结构:
struct edge 表示一条边,包括两个端点及其代价
edge graph[N] 表示有N条边组成的图
int father[N] 表示每个点的最上层的根节点
解释:因为这里需要判断是否形成环路,可以这样,每添加一条
边,看两个点是否在已经添加进去的边的点集中,若对需要添加
的这条边,发现两个点都在之前的那个集合中,这一定会形成回
路,所以,这里设置一个数组father[N],起初时,每个值为-1,代
表每个点的根节点都没有(因为没有添加一条边进去),当添加一条
边后,如果他们的根节点不同,则设置大的那个点的父节点为小
的那个点,如x > y 则 father[x] = y,这样每个点都只有一个根,
或者没有根,为-1,所以对添加进的节点,都可以查出他的根,然后
做比较,都相同,说明已位于添加进的节点中了,否则把该边添加
进去。 */ //定义一条边
struct edge{
int u; //起始点
int v; //目的点
int cost; //两点之间的代价
}; //这是一个对块数排序算法调用的一个比较函数
bool cmp(const edge &a, const edge &b){
return a.cost < b.cost;
} //查找一个节点的根节点
int findFather(int father[], int x){
//如果他的父节点不为-1,则应该递归,直到找到其父节点
if(father[x] != -1){
//将沿途的所有节点都指向同一个根节点
return father[x] = findFather(father, father[x]);
} //若为-1,则该点就是根
return x;
} //添加一条边
bool unionEdge(int father[], int x, int y){
//找到一条边的两个端点的根节点
x = findFather(father, x);
y = findFather(father, y); //根节点相同,说明已经加入了,再加入该边
//则会形成回路,该边舍弃,返回fasle
if(x == y){
return false;
} //若不同,让大的节点的根节点指向小的节点
if(x > y) father[x] = y;
if(x < y) father[y] = x; //该边可以加入,返回true
return true;
} int main(){
edge graph[N]; //定义了一个包含N条边的图
int father[N]; //定义了一个包含N个节点的根节点
int i,j, n; //n代表节点数
cin>>n;
//初始化数组
memset(graph, 0, sizeof(graph));
//初始化为-1表示任何点都没有父节点,即没有一条边已加入
memset(father, -1, sizeof(father)); int k = 0, cost, temp; //接收数据
for(i = 0;i < n;i++)
for(j = 0;j < n;j++){
if(i > j){
graph[k].u = i;
graph[k].v = j;
cin>>cost;
//对于小于0的值,表示不可达,所以代价为无穷大INF
if(cost < 0){
graph[k].cost = INF;
} else {
graph[k].cost = cost;
}
k++;
continue;
}
//由于是对称的,该值无用,但得接收
cin>>temp;
} //将所有边从小到大排序
sort(graph, graph + k, cmp); //打印排序后的边
for(i = 0;i < k;i++){
cout<<i<<" "<<graph[i].u<<"->"<<graph[i].v<<": "<<graph[i].cost<<endl;
} //count为记录已经加入的边数,到n-1时截止
//sum为最小生成树的代价和
int count = 0, sum = 0; //从小到大遍历k条边
for(i = 0; i < k;i++){
//探测该边是否可加入
if(unionEdge(father, graph[i].u, graph[i].v)){
count++;
sum += graph[i].cost;
} //当加入n-1条边时,已满足连通图,则退出
if(count == n - 1) break;
} cout<<"最小生成树代价和sum : "<<sum<<endl; return 0;
}
测试例子:
7
0 5 -1 -1 -1 11 2
5 0 10 8 -1 -1 13
-1 10 0 7 -1 -1 -1
-1 8 7 0 12 9 4
-1 -1 -1 12 0 10 -1
11 -1 -1 9 10 0 3
2 13 -1 4 -1 3 0
结果:
0 6->0: 2
1 6->5: 3
2 6->3: 4
3 1->0: 5
4 3->2: 7
5 3->1: 8
6 5->3: 9
7 2->1: 10
8 5->4: 10
9 5->0: 11
10 4->3: 12
11 6->1: 13
12 2->0: 1000000
13 6->4: 1000000
14 6->2: 1000000
15 3->0: 1000000
16 5->2: 1000000
17 5->1: 1000000
18 4->2: 1000000
19 4->1: 1000000
20 4->0: 1000000
最小生成树代价和sum : 31
Kruskal算法模拟讲解的更多相关文章
- Dijkstra算法模拟讲解
dijkstra算法,是一个求单源最短路径算法 其算法的特点为: 层层逼进,有点类似宽度搜索的感觉 其需要的数据结构为: int map[N][N] 所有点之间的权表 ...
- 最小生成树详细讲解(一看就懂!) & kruskal算法
0.前言 因为本人太蒟了 我现在连NOIP的初赛都在胆战心惊 并且我甚至连最小生成树都没有学过 所以这一篇博客一定是最详细的QAQ 哈哈 请您认真看完如果有疏漏之处敬请留言指正 感谢! Thanks♪ ...
- 最小生成树(prime算法 & kruskal算法)和 最短路径算法(floyd算法 & dijkstra算法)
一.主要内容: 介绍图论中两大经典问题:最小生成树问题以及最短路径问题,以及给出解决每个问题的两种不同算法. 其中最小生成树问题可参考以下题目: 题目1012:畅通工程 http://ac.jobdu ...
- POJ 1679 The Unique MST (次小生成树kruskal算法)
The Unique MST 时间限制: 10 Sec 内存限制: 128 MB提交: 25 解决: 10[提交][状态][讨论版] 题目描述 Given a connected undirect ...
- 图的最小生成树的理解和实现:Prim和Kruskal算法
最小生成树 一个连通图的生成树是一个极小的连通子图,它含有图中所有的顶点,但只有足以构成一棵树的n-1条边.我们将构造连通网的最小代价生成树称为最小生成树(Minimum Cost Spanning ...
- 最小生成树(II)与Kruskal算法
为防止网页加载过慢,故分两章.上接https://www.cnblogs.com/Uninstalllingyi/p/10479470.html Kruskal算法——将森林合并成树 玩过瘟疫公司吗… ...
- 算法学习记录-图——最小生成树之Kruskal算法
之前的Prim算法是基于顶点查找的算法,而Kruskal则是从边入手. 通俗的讲:就是希望通过 边的权值大小 来寻找最小生成树.(所有的边称为边集合,最小生成树形成的过程中的顶点集合称为W) 选取边集 ...
- 图论之最小生成树之Kruskal算法
Kruskal算法,又称作为加边法,是配合并查集实现的. 图示: 如图,这是一个带权值无向图我们要求它的最小生成树. 首先,我们发现在1的所有边上,连到3的边的边权值最小,所以加上这条边. 然后在3上 ...
- hdu 1875 畅通project再续(kruskal算法计算最小生成树)
畅通project再续 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Tota ...
随机推荐
- 解决VS2015无法调试dotnet core项目
dotnet core 1.0正式版和VS2015 update3安装后一直无法在VS中正常调试. 错误提示:The debugger's worker process (msvsmon.exe) u ...
- JS面向对象思想(OOP)
直接看js好了,模拟创建一个奥运会 function 奥运会Class(主题) { // 删除主题 // delete this.主题; this.主题 = 主题; this.开幕时间; this.闭 ...
- Linux Kernel Makefile Test
一.本文说明 本文为linux内核Makefile整体分析的续篇,是依据Linux内核Makefile体系的主要内容编写一个简要的测试工程.Linux内核Makefile体系就好像一只“大鸟”,而这篇 ...
- css详解笔记
CSS中的块级元素与行内元素 块级元素特性: 1.占据一整行,总是重起一行并且后面的元素也必须另起一行显示. 2.内联元素特性: 3.和其他内联元素显示在同一行. 块级元素列举如下: div(文档分区 ...
- iReport中求和的问题
数据库取出值TAX_AMT,但是不想在数据库里面计算,太麻烦,后面group by 字段太多.那就放到ireport里面去计算咯 在字段的如下位置进行计算吧.
- JBoss for luna
Redhat官方手顺 貌似已经404了,搬运地址 我的Eclipse是Luna,每次进入Eclipse Market去装 http://marketplace.eclipse.org/content/ ...
- Android程序的隐藏与退出
转自Android程序的隐藏与退出 Android的程序无需刻意的去退出,当你一按下手机的back键的时候,系统会默认调用程序栈中最上层Activity的Destroy()方法来销毁当前Activit ...
- codeforces D. Multiplication Table
http://codeforces.com/contest/448/problem/D 题意:一个n×m的矩阵,a[i][j]=i*j; 然后把a数组排序,找出第k个数. 思路:1-n×m二分枚举,然 ...
- SpringMVC+Json构建基于Restful风格的应用(转)
一.spring 版本:spring-framework-3.2.7.RELEASE 二.所需其它Jar包: 三.主要代码: web.xml <?xml version="1.0&qu ...
- lc面试准备:Regular Expression Matching
1 题目 Implement regular expression matching with support for '.' and '*'. '.' Matches any single char ...