日日算法:Kruskal算法
介绍
克鲁斯卡尔(Kruskal)算法是用来求出连通图中最小生成树的算法。
连通图:指无向图中任意两点都能相通的图。
最小生成树:指联通图的所有生成树中边权重的总和最小的树(即,找出一个树,让其联通所有的点,并让树的边权和为最小)。
算法思想
克鲁斯卡尔算法的主要基本思想有两点原则:
- 按照从小到大的顺序选择边,并将边的两端连线,构成新的图
- 保证新加入的边不能在新的图上形成环
- 重复以上步骤,直至添加
n-1条边
用图表示该算法的解体过程:

算法证明
我是通过反证的方式理解该算法的。
- 证明按上述算法添加
n-1条边时,一定能连通n个节点。
证明:
Kruskal算法保证了针对n个节点,它添加了n-1条边,且不存在环。那么假设这n-1条边没有全部连通n个节点。也就说至少有1个点没有边,那么至多只有n-1个点使用n-1条边,当n-1个点使用n-1条边时,必定构成环。与要求不符,故反正成立。
- 证明新的图中再添加一条边,一定构成环。
步骤一证明了新生成的图一定是一个连通图,也就是任意两点之间必定已经相连,当我们在加入一条新的边的时候,边两段的点又多了一条新相连的路,因此构成了环。
- 证明在构成新的环中,新加入的边一定是最长的边。
假设新加入的边,并非是环中最大的边,那么可以去掉这个环中最大的边,且剩下的边不够环,这与逐步加入小的边且不构成环这一条件矛盾。因此证明新加入的边一定为最长的边。
算法实现
public class Kruskal {
public static void generateMinTree(int[][] graph){
if(graph == null || graph.length <=0)
throw new IllegalArgumentException();
int minSum = 0;
//标记哪些点已经到访过
int[][] visited = new int[graph.length][graph.length];
//用来表示父子级的关系,验证是否存在环
int[] nodeHierarchy = new int[graph.length];
for(int i=0; i<nodeHierarchy.length; i++){
nodeHierarchy[i] = i;
}
int n = 0;
while(n < graph.length -1){
int minVal = Integer.MAX_VALUE;
int iIndex = 0;
int jIndex = 0;
for(int i=0; i<graph.length; i++){
for(int j=i+1; j<graph[i].length; j++){
if(graph[i][j] != Integer.MAX_VALUE && visited[i][j] == 0 && graph[i][j] < minVal){
iIndex = i;
jIndex = j;
minVal = graph[i][j];
}
}
}
visited[iIndex][jIndex] = 1;
//判断父节点是否相同,确定是否构成了环
if(findFather(nodeHierarchy, iIndex) != findFather(nodeHierarchy, jIndex)){
System.out.println(n + " Round min value path: " + minVal + " from " + iIndex + " to " + jIndex);
minSum += graph[iIndex][jIndex];
updateHierarchy(nodeHierarchy, iIndex, jIndex);
n++;
}
System.out.println("node hierarchy:" + Arrays.toString(nodeHierarchy));
}
System.out.println("min tree path sum:" + minSum);
System.out.println("node hierarchy:" + Arrays.toString(nodeHierarchy));
}
//递归查找父节点
private static int findFather(int[] nodeHierarchy, int idx){
if(nodeHierarchy[idx] == idx)
return idx;
return findFather(nodeHierarchy, nodeHierarchy[idx]);
}
//递归更新父节点
private static void updateHierarchy(int[] nodeHierarchy, int from, int to){
if(nodeHierarchy[from] != from)
updateHierarchy(nodeHierarchy, nodeHierarchy[from], from);
nodeHierarchy[from] = to;
}
}
上述代码见Github。
日日算法:Kruskal算法的更多相关文章
- 最小生成树之算法记录【prime算法+Kruskal算法】【模板】
首先说一下什么是树: 1.只含一个根节点 2.任意两个节点之间只能有一条或者没有线相连 3.任意两个节点之间都可以通过别的节点间接相连 4.除了根节点没一个节点都只有唯一的一个父节点 5.也有可能是空 ...
- 最小生成树(prime算法 & kruskal算法)和 最短路径算法(floyd算法 & dijkstra算法)
一.主要内容: 介绍图论中两大经典问题:最小生成树问题以及最短路径问题,以及给出解决每个问题的两种不同算法. 其中最小生成树问题可参考以下题目: 题目1012:畅通工程 http://ac.jobdu ...
- 最小生成树(Prim算法+Kruskal算法)
什么是最小生成树(MST)? 给定一个带权的无向连通图,选取一棵生成树(原图的极小连通子图),使生成树上所有边上权的总和为最小,称为该图的最小生成树. 求解最小生成树的算法一般有这两种:Prim算法和 ...
- [贪心经典算法]Kruskal算法
Kruskal算法的高效实现需要一种称作并查集的结构.我们在这里不介绍并查集,只介绍Kruskal算法的基本思想和证明,实现留在以后讨论. Kruskal算法的过程: (1) 将全部边按照权值由小到大 ...
- hdu 1233 还是畅通工程 最小生成树(prim算法 + kruskal算法)
还是畅通工程 Time Limit: 4000/2 ...
- 最小生成树 Prim算法 Kruskal算法实现
最小生成树定义 最小生成树是一副连通加权无向图中一棵权值最小的生成树. 在一给定的无向图 G = (V, E) 中,(u, v) 代表连接顶点 u 与顶点 v 的边(即,而 w(u, v) 代表此边的 ...
- 最小生成树Prim算法 Kruskal算法
Prim算法(贪心策略)N^2 选定图中任意定点v0,从v0开始生成最小生成树 树中节点Va,树外节点Vb 最开始选一个点为Va,其余Vb, 之后不断加Vb到Va最短距离的点 1.初始化d[v0]=0 ...
- 【431】Prim 算法 & Kruskal 算法
Prim 算法: Minimum Spanning Tree(MST):最小生成树,就是连接所有节点的最小权值 mst集合与rest集合 mst集合中顶点,找到一条最小权值的边 然后把边相关的顶点,选 ...
- 最小生成树Prim算法Kruskal算法
Prim算法采用与Dijkstra.Bellamn-Ford算法一样的“蓝白点”思想:白点代表已经进入最小生成树的点,蓝点代表未进入最小生成树的点. 算法分析 & 思想讲解: Prim算法每次 ...
- 克鲁斯卡尔算法(Kruskal算法)求最小生成树
题目传送:https://loj.ac/p/10065 1.排序函数sort,任何一种排序算法都行,下面的示例代码中,我采用的是冒泡排序算法 2.寻源函数getRoot,寻找某一个点在并查集中的根,注 ...
随机推荐
- C#通用类库整理--日志记录
日志的记录是将程序过程中的一些行为数据记录下来,方便开发.运维迅速的找到问题的所在,节省时间.使用时在 站点的web.config 中的<appSettings></appSetti ...
- VS2019中QT连接及使用
23:27:43 2019-08-09 qt连接VS 连接前提是在下载qt的时候将 MSVC 2017装上 点击扩展 选择管理扩展 搜索qt 选择下载 之后下载结束并重新打开后 会弹出一个 QT o ...
- Vulnhub DC-6靶机渗透
信息搜集 nmap -sP 192.168.146.0/24 #找靶机ip nmap -sS -Pn -A 192.168.146.143 #扫描靶机信息 22和80端口,老朋友了. 先直接访问htt ...
- 路由与交换,cisco路由器配置,静态路由
网络是一个大型的拓扑结构,在路由表中,最重要的是管理距离和度量值 管理距离 管理距离用来确定路由的优先级.管理距离的范围是0-255之间的整数值.值越低代表优先级越高.0代表最高优先级.并且只有直连路 ...
- 23 抽象类 abstract
/*概念 * abstract:关键字,用于修饰方法和类 * 抽象方法:不同类的方法是相似,但是具体内容又不太一样,所以我们只能抽取他的声明,没有具体的方法体,没有具体方法体的方法就是抽象方法 * 抽 ...
- hadoop(五)scp命令copy文件和配置(完全分布式准备二)|7
机器的克隆参考centos7克隆ip|机器名|映射关系|别名配置(hadoop完全分布式准备一) 那么问题来了,如果我们有30台机器,之间可以互相访问,那我们如何快速安装配置环境如jdk hadoop ...
- 【Selenium06篇】python+selenium实现Web自动化:日志处理
一.前言 最近问我自动化的人确实有点多,个人突发奇想:想从0开始讲解python+selenium实现Web自动化测试,请关注博客持续更新! 这是python+selenium实现Web自动化第六篇博 ...
- pgsql中json格式数组查询结果变成了字符串
场景复原 最近使用到了json的数组,用来存储多个文件的值,发现在连表查询的时候返回结果变成了字符串. { "id": "repl-placeholder-007&quo ...
- 原创Pig0.16.0安装搭建
tar -zxvf pig-0.16.0.tar.gz -C ~ vi ~/.bash_profile export PIG_HOME=/home/hadoop/pig-0.16.0 export ...
- linux通过进程名查看其占用端口
1.先查看进程pid ps -ef | grep 进程名 2.通过pid查看占用端口 netstat -nap | grep 进程pid 参考: https://blog.csdn.net/sinat ...