最小生成树——Prim算法理解
背景:本文是在小甲鱼数据结构教学视频中的代码的基础上,添加详细注释而完成的。该段代码并不完整,仅摘录了核心算法部分,结合自己的思考,谈谈理解。
Prim算法理解:
如图(摘录自小甲鱼教学视频中的图片),是一个带有权值的连通网。
根据上图可以列写出该连通网的邻接矩阵,为了方便直观的理解:
权值 | V0 | V1 | V2 | V3 | V4 | V5 |
V0 | 0 | 6 | 1 | 5 | Ꚙ | Ꚙ |
V1 | 6 | 0 | 5 | Ꚙ | 3 | Ꚙ |
V2 | 1 | 5 | 0 | 7 | 5 | 4 |
V3 | 5 | Ꚙ | 7 | 0 | Ꚙ | 2 |
V4 | Ꚙ | 3 | 5 | Ꚙ | 0 | 6 |
V5 | Ꚙ | Ꚙ | 4 | 2 | 6 | 0 |
如上图所示,实际上Prim算法求解最小生成树的过程就是不断地寻找已走过顶点的最近顶点的过程(边权值最小)。
因此如上图所示,起点在V0点,寻找最近顶点,即V2点;找到V2点后,V2点作为当前顶点,继续寻找最近顶点,这时和已走过的顶点(V0、V2)直接连通的点共有4个:V1、V3、V4、V5,但是显然求解最小生成树不能重复进入已经走过的点,对于这个问题Prim算法采用的方法是将已经走过的点到任何点的权值都设定为0,然后在寻找最短边时,忽略权值为0点边。
这样当我们站在V2点处时,已走过的点到各个点之间的最短权值分别为:
权值 | V0 | V1 | V2 | V3 | V4 | V5 |
V0、V2 | 0 | 5 | 0 | 5 | 5 | 4 |
以上叙述的内容其实就是Prim算法的核心,即:①寻找最近点;②迭代lowcost数组。
在实际的算法实现中,就是通过这样一步一步迭代lowcost数组,然后在lowcost数组中寻找最小的权值,作为目标点,最终实现以最短路径遍历连通网。
代码及详细注释:
void MiniSpanTree_Prim(MGraph G)
{
int min,i,j,k;
/* 邻接数组 保存每一个顶点的邻接点(也就是上一个走过的点)的下标 */
int adjvex[MAXWEX];
/* 保存当前顶点向其他顶点的距离(其中已完成的顶点置零,表示不能再走) */
int lowcost[MAXWEX];
lowcost[0] = 0;
adjvex[0] = 0;
/* init */
for(i=1;i<G.numVertexes;i++)
{
/* 将邻接矩阵第0行所有权值先加入数组 */
/* 也就是将生成树的初始位置设定在该顶点 */
/* 注意这个lowcost数组是逐级更新的:迭代到下一个点的时候,更新locwost数组 */
/* lowcost数组更新原则: 已走过的点的lowcost值均保持0(即不再重复进入)
将当前点到其他点的权值小于lowcost数组的,更新到lowcost数组中(为了保证每一次走的都是最短路径) */
lowcost[i] = G.arc[0][i];
/* 邻接数组初始化为0 */
adjvex[i] = 0;
}
/* the real process creating the minispantree */
for(i=1;i<G.numVertexes;i++)
{
/* min初始化为不可能的值(65535) */
min = INFINITY;
j = 1;
k = 0;
/* 遍历全部顶点 */
while(j<G.numVertexes)
{
/* 找到当前顶点的可行边中权值最小的边 此即当前顶点的下一个邻接点 */
if(lowcost[j] != 0 && lowcost[j] < min)
{
min = lowcost[j];
k = j;
}
j++;
}
/* 打印点与点的邻接关系 遍历完成后这些邻接关系组合在一起就是最小生成树 */
printf("(%d,%d)",adjvex[k],k);
/* 这里很重要 */
/* 将已经走过的顶点对应lowcost数组中的权值置0 已确保不再重复进入 */
lowcost[k] = 0;
/* for循环完成lowcost数组的更新(更新原则在lowcost数组的定义初有详细注释) */
for(j=1;j<G.numVertexes;j++)
{
/* 到已走过的点的权值不更新(保持为0,两行之前的赋值完成) */
/* 邻接矩阵中当前点到其他点的权值小于lowcost数组的更新 */
if(lowcost[j] != 0 && G.arc[k][j] < lowcost[j])
{
lowcost[j] = G.arc[k][j];
/* 邻接数组赋值(记录各点之间的邻接关系) */
adjvex[j] = k;
}
}
}
}
——cloud over sky
——2020/3/11
最小生成树——Prim算法理解的更多相关文章
- 最小生成树——Kruskal算法理解
背景:本文是在小甲鱼数据结构教学视频中的代码的基础上,添加详细注释而完成的.该段代码并不完整,仅摘录了核心算法部分,结合自己的思考,谈谈理解. Prim算法理解: 如图(摘录自小甲鱼教学视频中的图片) ...
- 图论算法(五)最小生成树Prim算法
最小生成树\(Prim\)算法 我们通常求最小生成树有两种常见的算法--\(Prim\)和\(Kruskal\)算法,今天先总结最小生成树概念和比较简单的\(Prim\)算法 Part 1:最小生成树 ...
- 最小生成树,Prim算法与Kruskal算法,408方向,思路与实现分析
最小生成树,Prim算法与Kruskal算法,408方向,思路与实现分析 最小生成树,老生常谈了,生活中也总会有各种各样的问题,在这里,我来带你一起分析一下这个算法的思路与实现的方式吧~~ 在考研中呢 ...
- 数据结构代码整理(线性表,栈,队列,串,二叉树,图的建立和遍历stl,最小生成树prim算法)。。持续更新中。。。
//归并排序递归方法实现 #include <iostream> #include <cstdio> using namespace std; #define maxn 100 ...
- 最小生成树Prim算法(邻接矩阵和邻接表)
最小生成树,普利姆算法. 简述算法: 先初始化一棵只有一个顶点的树,以这一顶点开始,找到它的最小权值,将这条边上的令一个顶点添加到树中 再从这棵树中的所有顶点中找到一个最小权值(而且权值的另一顶点不属 ...
- 最小生成树—prim算法
最小生成树prim算法实现 所谓生成树,就是n个点之间连成n-1条边的图形.而最小生成树,就是权值(两点间直线的值)之和的最小值. 首先,要用二维数组记录点和权值.如上图所示无向图: int map[ ...
- Highways POJ-1751 最小生成树 Prim算法
Highways POJ-1751 最小生成树 Prim算法 题意 有一个N个城市M条路的无向图,给你N个城市的坐标,然后现在该无向图已经有M条边了,问你还需要添加总长为多少的边能使得该无向图连通.输 ...
- SWUST OJ 1075 求最小生成树(Prim算法)
求最小生成树(Prim算法) 我对提示代码做了简要分析,提示代码大致写了以下几个内容 给了几个基础的工具,邻接表记录图的一个的结构体,记录Prim算法中最近的边的结构体,记录目标边的结构体(始末点,值 ...
- 模板——最小生成树prim算法&&向前星理解
通过最小生成树(prim)和最短路径优化引出的向前星存图,时至今日才彻底明白了.. head[i]存储的是父节点为i引出的最后一条边的编号, next负责把head[i]也就是i作为父节点的所有边连接 ...
随机推荐
- libevent(七)信号事件监听
libevent通过socketpair实现对信号事件的监听. 还记得event_base吗? struct event_base { struct evsig_info sig; // 略 }; e ...
- 集成学习基础知识总结-Bagging-Boosting
理论 在概率近似正确((probably approximately correct)学习框架下.一个概念是强可学习的充分必要条件是这个概念是弱可学习(仅比随机猜测稍好). 要求 个体学习器要好而不同 ...
- Lasso回归
Lasso 是一个线性模型,它给出的模型具有稀疏的系数(sparse coefficients).它在一些场景中是很有用的,因为它倾向于使用较少参数的情况,能够有效减少给定解决方案所依赖变量的个数.因 ...
- Linux内核驱动学习(七)应用层直接操作GPIO
文章目录 简介 原理图 节点 设置为输出 设置为输入 映射关系 debugfs pwm demo 简介 前面通过libgpio的方式介绍了内核空间对GPIO进行操作的接口,其做了较好的封装,同时Lin ...
- Linux 内核工作队列之work_struct 学习总结
前言 编写Linux驱动的时候对于work_struct的使用还是很普遍的,很早之前就在阅读驱动源码的时候就看到了它的踪影,根据其命名大概知道了它的具体作用,但是仍然不知所以,同时,伴随出现的还有de ...
- [CodeForces 300D Painting Square]DP
http://codeforces.com/problemset/problem/300/D 题意:每一次操作可以选一个正方形,令边长为n,如果n为奇数那么可以从中间画一个十字,分成4个大小相等的边长 ...
- Spark2.4.5集群安装与本地开发
下载 官网地址:https://www.apache.org/dyn/closer.lua/spark/spark-2.4.5/spark-2.4.5-bin-hadoop2.7.tgz 验证Java ...
- 08JAVA基础关键字(final、static)以及抽象类和接口
一.关键字 1.final 修饰类 修饰变量 修饰成员方法 该类为最终类,不能被继承 该变量为常量 该成员方法不能被重写 2.static (1).生命周期 随着类的加载而加载 (2).特点 被本类所 ...
- void 型指针的高阶用法,你掌握了吗?
[导读] 要比较灵活的使用C语言实现一些高层级的框架时,需要掌握一些进阶编程技巧,这篇来谈谈void指针的一些妙用.测试环境采用 IAR for ARM 8.40.1 什么是void指针 void指针 ...
- Spring Boot集成H2数据库
需求 平时学习的时候,涉及到一些连接数据库相关的操作,经常需要初始化本地数据库,比如装个MySQL,初始化一些脚本,比较麻烦,H2是内存数据库,Spring Boot可以在应用启动的时候对H2数据库初 ...