最小生成树

最小生成树(minimum spanning tree)是由n个顶点,n-1条边,将一个连通图连接起来,且使权值最小的结构。
最小生成树可以用Prim(普里姆)算法或kruskal(克鲁斯卡尔)算法求出。

我们将以下面的带权连通图为例讲解这两种算法的实现:

注:由于测试输入数据较多,程序可以采用文件输入



Prim(普里姆)算法
时间复杂度:O(N^2)(N为顶点数)
prim算法又称“加点法”,用于边数较多的带权无向连通图
方法:每次找与之连线权值最小的顶点,将该点加入最小生成树集合中
注意:相同权值任选其中一个即可,但是不允许出现闭合回路的情况。






代码部分通过以下步骤可以得到最小生成树:

1.初始化:
lowcost[i]:表示以i为终点的边的最小权值,当lowcost[i]=0表示i点加入了MST。
mst[i]:表示对应lowcost[i]的起点,当mst[i]=0表示起点i加入MST。
由于我们规定最开始的顶点是1,所以lowcost[1]=0,MST[1]=0。即只需要对2~n进行初始化即可。

#define MAX 100
#define MAXCOST 0x7fffffff  int graph[MAX][MAX];  void prim(int graph[][MAX], int n)
{  int lowcost[MAX];  int mst[MAX];  int i, j, min, minid, sum = 0;  for (i = 2; i <= n; i++)  {  lowcost[i] = graph[1][i];//lowcost存放顶点1可达点的路径长度 mst[i] = 1;//初始化以1位起始点 }  mst[1] = 0;

2.查找最小权值及路径更新
定义一个最小权值min和一个最小顶点ID minid,通过循环查找出min和minid,另外由于规定了某一顶点如果被连入,则lowcost[i]=0,所以不需要担心重复点问题。所以找出的终点minid在MST[i]中可以找到对应起点,min为权值,直接输出即可。
我们连入了一个新的顶点,自然需要对这一点可达的路径及权值进行更新,所以循环中还应该包括路径更新的代码。

for (i = 2; i <= n; i++)  {  min = MAXCOST;  minid = 0;  for (j = 2; j <= n; j++)  {  if (lowcost[j] < min && lowcost[j] != 0)  {  min = lowcost[j];//找出权值最短的路径长度 minid = j; //找出最小的ID }  }  printf("V%d-V%d=%d\n",mst[minid],minid,min); sum += min;//求和 lowcost[minid] = 0;//该处最短路径置为0 for (j = 2; j <= n; j++){  if (graph[minid][j] < lowcost[j])//对这一点直达的顶点进行路径更新 {  lowcost[j] = graph[minid][j];  mst[j] = minid;}  }  }  printf("最小权值之和=%d\n",sum);
}

具体代码如下:

#include<stdio.h>
#define MAX 100
#define MAXCOST 0x7fffffff  int graph[MAX][MAX];  void prim(int graph[][MAX], int n)
{  int lowcost[MAX];  int mst[MAX];  int i, j, min, minid, sum = 0;  for (i = 2; i <= n; i++)  {  lowcost[i] = graph[1][i];//lowcost存放顶点1可达点的路径长度 mst[i] = 1;//初始化以1位起始点 }  mst[1] = 0;  for (i = 2; i <= n; i++)  {  min = MAXCOST;  minid = 0;  for (j = 2; j <= n; j++)  {  if (lowcost[j] < min && lowcost[j] != 0)  {  min = lowcost[j];//找出权值最短的路径长度 minid = j; //找出最小的ID }  }  printf("V%d-V%d=%d\n",mst[minid],minid,min); sum += min;//求和 lowcost[minid] = 0;//该处最短路径置为0 for (j = 2; j <= n; j++){  if (graph[minid][j] < lowcost[j])//对这一点直达的顶点进行路径更新 {  lowcost[j] = graph[minid][j];  mst[j] = minid;}  }  }  printf("最小权值之和=%d\n",sum);
}
int main()
{  int i, j, k, m, n;  int x, y, cost;  //freopen("1.txt","r",stdin);//文件输入 scanf("%d%d",&m,&n);//m=顶点的个数,n=边的个数  for (i = 1; i <= m; i++)//初始化图 {  for (j = 1; j <= m; j++)  {  graph[i][j] = MAXCOST;  }  }   for (k = 1; k <= n; k++)  {  scanf("%d%d%d",&i,&j,&cost);graph[i][j] = cost;  graph[j][i] = cost;  }  prim(graph, m);  return 0;
}

编译运行结果:



kruskal(克鲁斯卡尔)算法
时间复杂度:O(NlogN)(N为边数)
kruskal算法又称“加边法”,用于边数较少的稀疏图
方法:每次找图中权值最小的边,将边连接的两个顶点加入最小生成树集合中
注意:相同权值任选其中一个即可,但是不允许出现闭合回路的情况。






代码部分通过以下步骤可以得到最小生成树:

1.初始化:
构建边的结构体,包括起始顶点、终止顶点,边的权值
借用一个辅助数组vset[i]用来判断某边是否加入了最小生成树集合

#define MAXE 100
#define MAXV 100
typedef struct{int vex1;                     //边的起始顶点int vex2;                      //边的终止顶点int weight;                    //边的权值
}Edge;
void kruskal(Edge E[],int n,int e)
{ int i,j,m1,m2,sn1,sn2,k,sum=0;int vset[n+1];for(i=1;i<=n;i++)        //初始化辅助数组vset[i]=i;k=1;//表示当前构造最小生成树的第k条边,初值为1j=0;//E中边的下标,初值为0

2.取边和辅助集合更新
按照***排好的顺序***依次取边,若不属于同一集合则将其加入最小生成树集合,每当加入新的边,所连接的两个点即纳入最小生成树集合,为避免重复添加,需要进行辅助集合更新
注:由于kruskal算法需要按照权值大小顺序取边,所以应该事先对图按权值升序,这里我采用了快速排序算法,具体算法可以参照快速排序(C语言)

 while(k<e)//生成的边数小于e时继续循环{m1=E[j].vex1;m2=E[j].vex2;//取一条边的两个邻接点sn1=vset[m1];sn2=vset[m2];                           //分别得到两个顶点所属的集合编号if(sn1!=sn2)//两顶点分属于不同的集合,该边是最小生成树的一条边{//防止出现闭合回路 printf("V%d-V%d=%d\n",m1,m2,E[j].weight);sum+=E[j].weight;k++;                //生成边数增加if(k>=n)break;for(i=1;i<=n;i++)    //两个集合统一编号if (vset[i]==sn2)  //集合编号为sn2的改为sn1vset[i]=sn1;}j++;                  //扫描下一条边}printf("最小权值之和=%d\n",sum);
}

具体算法实现:

#include <stdio.h>
#define MAXE 100
#define MAXV 100
typedef struct{int vex1;                     //边的起始顶点int vex2;                      //边的终止顶点int weight;                    //边的权值
}Edge;
void kruskal(Edge E[],int n,int e)
{ int i,j,m1,m2,sn1,sn2,k,sum=0;int vset[n+1];for(i=1;i<=n;i++)        //初始化辅助数组vset[i]=i;k=1;//表示当前构造最小生成树的第k条边,初值为1j=0;//E中边的下标,初值为0while(k<e)//生成的边数小于e时继续循环{m1=E[j].vex1;m2=E[j].vex2;//取一条边的两个邻接点sn1=vset[m1];sn2=vset[m2];                           //分别得到两个顶点所属的集合编号if(sn1!=sn2)//两顶点分属于不同的集合,该边是最小生成树的一条边{//防止出现闭合回路 printf("V%d-V%d=%d\n",m1,m2,E[j].weight);sum+=E[j].weight;k++;                //生成边数增加 if(k>=n)break;for(i=1;i<=n;i++)    //两个集合统一编号if (vset[i]==sn2)  //集合编号为sn2的改为sn1vset[i]=sn1;}j++;                  //扫描下一条边}printf("最小权值之和=%d\n",sum);
}
int fun(Edge arr[],int low,int high){int key;Edge lowx;lowx=arr[low];key=arr[low].weight;while(low<high){while(low<high && arr[high].weight>=key)high--;if(low<high)arr[low++]=arr[high];while(low<high && arr[low].weight<=key)low++;if(low<high)arr[high--]=arr[low];}arr[low]=lowx;return low;}
void quick_sort(Edge arr[],int start,int end)
{int pos;if(start<end){pos=fun(arr,start,end);quick_sort(arr,start,pos-1);quick_sort(arr,pos+1,end);}
}
int main()
{Edge E[MAXE];int nume,numn;//freopen("1.txt","r",stdin);//文件输入printf("输入顶数和边数:\n");scanf("%d%d",&numn,&nume);for(int i=0;i<nume;i++)scanf("%d%d%d",&E[i].vex1,&E[i].vex2,&E[i].weight);quick_sort(E,0,nume-1);kruskal(E,numn,nume);
}

编译运行结果:

最小生成树构造算法--Prim算法,Kruskal算法(C语言)相关推荐

  1. 最小生成树(Prim、Kruskal)算法,秒懂!

    前言 在数据结构与算法的图论中,(生成)最小生成树算法是一种常用并且和生活贴切比较近的一种算法.但是可能很多人对概念不是很清楚,什么是最小生成树? 一个有 n 个结点的连通图的生成树是原图的极小连通子 ...

  2. dijkstra算法_Python实现图的经典DFS、BFS、Dijkstra、Floyd、Prim、Kruskal算法

    讲在前面的话,图的算法太多,理论知识肯定一篇文章讲不完,关于理论知识大家可以参考教材Sedgewick的<算法>或reference的链接,本文主要还是想在一篇文章中记录六种算法的Pyth ...

  3. dfs时间复杂度_Python实现图的经典DFS、BFS、Dijkstra、Floyd、Prim、Kruskal算法

    讲在前面的话,图的算法太多,理论知识肯定一篇文章讲不完,关于理论知识大家可以参考教材Sedgewick的<算法>或reference的链接,本文主要还是想在一篇文章中记录六种算法的Pyth ...

  4. 数据结构(六):图的概念、存储方式、基本操作、最小生成树、最短路径、有向无环图、关键路径 | Prim、Kruskal算法 | BFS、Dijkstra、Floyd算法 | 拓扑排序 | 求关键路径

    文章目录 第六章 图 一.图 (一)图的定义 (二)图逻辑结构的应用 (三)无向图.有向图 (四)简单图.多重图 (五)顶点的度.入度.出度 (六)顶点-顶点的关系描述 (七)连通图.强连通图 (八) ...

  5. Prim和Kruskal算法应用----城市水管连接

    Prim和Kruskal算法应用----城市水管连接 问题描述: Description: 现在有n个城镇,编号为1, 2, 3, 4-n.他们之间有m条互通的道路,每条道路有相应的长度,现在基于这些 ...

  6. 【数据结构】 最小生成树(四)——利用kruskal算法搞定例题×3+变形+一道大水题...

    在这一专辑(最小生成树)中的上一期讲到了prim算法,但是prim算法比较难懂,为了避免看不懂,就先用kruskal算法写题吧,下面将会将三道例题,加一道变形,以及一道大水题,水到不用高级数据结构,建 ...

  7. 【图解】Prim和Kruskal算法的区别

    [贪心]Prim和Kruskal算法的区别 Kruskal算法和Prim算法的优劣 Kruskal算法,相较于Prim算法是基于点的操作,Kruskal算法是基于边的操作,思想也比Prim简单,更容易 ...

  8. 【NOJ1596、1597】【贪心算法之最小生成树】最少修建多长的公路能把所有村庄连起来(图示Prim与Kruskal算法)

    1596.最少修建多长的公路能把所有村庄连起来(一) 时限:1000ms 内存限制:10000K  总时限:3000ms 描述 一个地区有n个村庄,有一些村子之间可以修路,已知每条路的长度,问最少修建 ...

  9. Prim和Kruskal算法

    1. Prim算法 1.1 概览 普里姆算法(Prim算法),图论中的一种算法,可在加权连通图里搜索最小生成树.意即由此算法搜索到的边子集所构成的树中,不但包括了连通图里的所有顶点(英语:Vertex ...

  10. 浙大数据结构:08-图7 公路村村通 (30 分)Prim与Kruskal算法

    08-图7 公路村村通 (30 分) 现有村落间道路的统计数据表中,列出了有可能建设成标准公路的若干条道路的成本,求使每个村落都有公路连通所需要的最低成本. 输入格式: 输入数据包括城镇数目正整数N( ...

最新文章

  1. java servlet 请求_Java中前台JSP请求Servlet实例(http+Servlet)
  2. jQuery技术内幕:深入解析jQuery架构设计与实现原理
  3. 2020-12-5(操作系统---设备管理)
  4. 更灵活的边缘云原生运维:OpenYurt 单元化部署新增 Patch 特性
  5. Frogs HDU - 5514
  6. 90TB显存!英伟达发布新一代SuperPod超算,AI算力新巅峰!
  7. C# Winform 窗体美化(一、IrisSkin 换肤库)
  8. 推荐一个很好的富文本web编辑器UEditor
  9. Webpack的使用——进阶篇
  10. C#获取本地IP地址
  11. 基于matlab的产生式系统案例
  12. python实现2000投影坐标转经纬度
  13. 4.20 扣1送地狱火
  14. uIP使用例子应用(一)
  15. wr720n刷成网络打印_TP-Link WR720N刷入OpenWrt之一般流程
  16. 小程序 微信位置详情页打开小程序
  17. 徐小湛概率论与数理统计课件_概率论与数理统计-徐小湛-视频教程70讲
  18. oracle 创建表 as,Oracle创建表(create table as)
  19. 【GIS人必学】零基础学习ArcGIS Python脚本开发训练营来了
  20. js和html5实现扫描条形码

热门文章

  1. 评估指标_机器学习评估方法与评估指标总结
  2. md5 java 工具类_Java实现一个MD5工具类
  3. linux进程运行队列,Linux进程调度中队列的使用
  4. python高频词_python几万条微博高频词分析
  5. 整型和浮点型之间的转化
  6. antd table排序 vue_antd of Vue 之table组件踩坑记
  7. 车仪表台上的装饰_一汽大众销量最差的车,旅行车蔚领为什么在国内没有市场?...
  8. excel工具箱_WPS搬来救兵智能工具箱,强大到Excel没有还手之力
  9. 数学建模学习笔记(九)——聚类模型
  10. Python 小白从零开始 PyQt5 项目实战(8)汇总篇(完整例程)