本文出自我的掘金博客, 欢迎大家访问 传送门

最小生成树的最著名的算法有两个, 一个是Prim算法, 另一个当然就是Kruskal算法, 接下来, 我将尽我所能的介绍这两个算法, 也算是对自己学习的一个回顾吧

老规矩, 模板题传送门

首先, 介绍我更喜欢的, 也是相对更容易敲代码的Kruskal算法

按照离散数学的定义

> 基本思想:按照权值从小到大的顺序选择n-1条边,并保证这n-1条边不构成回路。> 具体做法:首先构造一个只含n个顶点的森林,然后依权值从小到大从连通网中选择边加入到森
林中,并使森林中不产生回路,直至森林变成一棵树为止。

故我们可以提炼出算法的流程如下

1.将边升序排序
2.判断是否能插入此边,插入后做:1.inc(ans,路径长度)2.合并连通分支

1.对于排序, 可以借助于库函数sort

2.对于判断是否可以插入的步骤, 我们的想法是借助于并查集, 如果这条边的两个祖先相同, 那么插入这条边显然会构成环, 故跳过

3.算法结束的标志是加入的边的数目为n - 1

算法已经介绍清楚了, 那么下面我们就来考虑一下存储边的数据结构, 我们选择了如下所示的结构体数组

struct Edge {int u; //起点int v; //终点int w; //权值
} e[100010];

对于并查集的处理就简单的提一下

1. 初始化

void init() {for (int i = 1; i <= n; i++) {fa[i] = i;}
}

2.寻找祖先的函数, 路径压缩算法

int find (int x) {return fa[x] == x ? fa[x] : fa[x] = find(fa[x]);
}

3.合并操作, 将两个属于不同集合的顶点合并到同一个集合, 即入赘操作

void union_(int x, int y) {int fx = find(x);int fy = find(y);fa[fx] = fy;
}

感觉也没多少东西, 那就直接贴完整AC代码吧

#include <bits/stdc++.h>using namespace std;
const int MAXN = 100 + 10;
int n, cnt, fa[MAXN], sum, ans;void init() {for (int i = 1; i <= n; i++) {fa[i] = i;}
}int find (int x) {return fa[x] == x ? fa[x] : fa[x] = find(fa[x]);
}void union_(int x, int y) {int fx = find(x);int fy = find(y);fa[fx] = fy;
}struct Edge {int u;int v;int w;
} e[100010];bool cmp (Edge a, Edge b) {return a.w < b.w;
}
//这一题应该采用并查集来判断是否会构成一个环, 然后用一个结构体数组来存储边的信息
int main() {cin >> n;init();int h;for (int i = 1; i <= n; i++) {for (int j = 1; j <= n; j++) {cin >> h;if (j > i) {//存一半就可以了e[++cnt].u = i;e[cnt].v = j;e[cnt].w = h;}}}sort(e + 1, e + 1 + cnt, cmp);for (int i = 1; i <= cnt; i++) {int u = e[i].u;int v = e[i].v;if (find(u) != find(v)) {union_(u, v);sum++;ans += e[i].w;if (sum == n - 1) {break;}}}cout << ans;return 0;
}

二, Prime算法

Prim算法也是一个贪心算法, 它和Kruskal算法的区别在于Prim算法是每次从一个点出发选择当前点的不构成环的最小的边, 而Kruskal算法是从全局的角度, 从所有的边中选择不构成环的最小的边, 所以Prim算法相对而言复杂一些

很显然, 每次都要从当前的顶点选择最优的边, 那么这就是一个很耗时的操作, 于是我们可以对这一步进行堆优化(其实就是使用优先队列)

这个算法的数据结构相对复杂一些, 接下来我来介绍一下

bool vis[MAXN]; //用来判断这个顶点是否访问过struct Edge {int u; //起点int v; //终点int w; //权值bool operator <(const struct Edge& n) const {return w > n.w;} //重载比较运算符, 用于后面的优先队列
};vector <Edge> g[MAXN]; //向量数组, 用于存放每一个顶点所连接的边的信息
priority_queue <Edge> edge; //优先队列不解释

1.读取数据

for (int i = 1; i <= n; i++) {for (int j = 1; j <= n; j++) {int w;cin >> w;if (w == 0) continue;g[i].push_back(Edge{i, j, w});}
}

2.以1为起始点, 将其相连边入队

vis[1] = true;
for (int i = 0; i < g[1].size(); i++) {edge.push(g[1][i]);
}

3.执行Prim算法, 直到边数为n - 1为止

 while (cnt < n - 1){int w = edge.top().w;int v = edge.top().v;edge.pop();if (vis[v]) {//已经访问过了continue;}vis[v] = true;ans += w; //ans是结果cnt++; //cnt是边的计数器for (int i = 0; i < g[v].size(); i++) {if (!vis[g[v][i].v]) {edge.push(g[v][i]);}}
}

完整的代码就不贴了

好了, 今天就分享到这里

最小生成树算法超详细教程相关推荐

  1. 100个python算法超详细讲解:搬山游戏

    [100个python算法超详细讲解]@谷歌学术 1.问题描述 设有n座山,计算机与人作为比赛的双方,轮流搬山.规定每次搬 山数不能超过k座,谁搬最后一座谁输.游戏开始时,计算机请人输入 山的总数n和 ...

  2. 100个python算法超详细讲解:平分7筐鱼

    [100个python算法超详细讲解]@谷哥技术 1.问题描述 2.乙.丙三位渔夫出海打鱼,他们随船带了21只箩筐.当晚返航时, 他们发现有7筐装满了鱼,还有7筐装了半筐鱼,另外7筐是空的,由于他们 ...

  3. 100个python算法超详细讲解:邮票组合

    [100个python算法超详细讲解]@谷哥技术 1.问题描述 我们寄信都要贴邮票,在邮局有一些小面值的邮票,通过这些小 面值邮票中的一张或几张的组合,可以满足不同邮件的不同邮资.现 在,邮局有4种不 ...

  4. 100个python算法超详细讲解:递归解决年龄问题

    [100个python算法超详细讲解]@谷哥技术 1.问题描述 有5个人坐在一起,问第5个人多少岁,他说比第4个人大2岁.问 第4个人多少岁,他说比第3个人大2岁.问第3人多少岁,他说比第2个 人大2 ...

  5. 100个python算法超详细讲解:谁是窃贼

    [100个python算法超详细讲解]@谷哥技术 1.问题描述 警察审问4名窃贼嫌疑犯.现在已知,这4人当中仅有一名是窃 贼,还知道这4个人中的每个人要么是诚实的,要么总是说谎. 下面是这4个人给警察 ...

  6. 100个python算法超详细讲解:抢30游戏

    [100个python算法超详细讲解]@谷歌学术 1.问题描述 由两个人玩"抢30"游戏,游戏规则是:第一个人先说"1"或"1,2",第 二 ...

  7. 推荐系统[二]:召回算法超详细讲解[召回模型演化过程、召回模型主流常见算法(DeepMF/TDM/Airbnb Embedding/Item2vec等)、召回路径简介、多路召回融合]

    搜索推荐系统专栏简介:搜索推荐全流程讲解(召回粗排精排重排混排).系统架构.常见问题.算法项目实战总结.技术细节以及项目实战(含码源) 专栏详细介绍:搜索推荐系统专栏简介:搜索推荐全流程讲解(召回粗排 ...

  8. 100个python算法超详细讲解:卡布列克常数

    [100个python算法超详细讲解]@谷哥技术 1.问题描述 对于任意一个4位数n,进行如下的运算: 1)将组成该4位数的4个数字由大到小排列,形成由这4个数字构 成的最大的4位数. 2)将组成该4 ...

  9. 100个python算法超详细讲解:孪生素数

    [100个python算法超详细讲解@谷哥技术] 1.问题描述 本节要研究孪生素数的问题,先来看看什么是孪生素数. 所谓孪生素数指的是间隔为2的两个相邻素数,因为它们之间的距离已经 近得不能再近了,如 ...

最新文章

  1. Android众说纷纭分辨率
  2. python核心教程第二版答案_《python核心教程》课后题答案注解
  3. struts2访问session的两种方法
  4. 新手向:Vue 2.0 的建议学习顺序
  5. 实验1 LINUX基本操作
  6. python有强大吗_python有多强大
  7. asp.net上传图片文件自动修改图片大小代码
  8. 一文彻底掌握Apache Hudi异步Clustering部署
  9. 《软件工程》实验报告——软件测试
  10. LeetCode LCP 33. 蓄水(暴力枚举)
  11. Intel Hex概述
  12. python 如何修改字典的key值_Python编程:如何把元组/字典作为参数传递给函数(附代码)...
  13. c语言有趣的软件,一个有趣的小程序
  14. 一、配置etcd数据库
  15. 【图像计数】基于matlab灰度二值化同类物体简单计数【含Matlab源码 759期】
  16. python自动化交易通达信_通达信实现自动交易系统
  17. NATAPP使用教程(内网穿透)
  18. 【H5】 svg的 defs用法 渐变
  19. TongWeb卡、TongWeb卡、TongWeb卡卡卡
  20. MV* 模式梳理与理解(还原真实的 MV* 模式)

热门文章

  1. MFC实现交通图导航系统
  2. 一个人工作之余6元上线一款APP的血泪史
  3. win10:谷歌浏览器如何导出扩展程序
  4. win10 git bash 设置别名
  5. 3种在JavaScript中验证电子邮件地址的方法
  6. Qt中鼠标的双击事件和单击事件的实现方式
  7. java font 字体加粗_java字体设置,包括大小,颜色,加粗,下划线,对齐,斜体的设置...
  8. win11怎么看电脑显卡信息
  9. mysql综训实训报告总结_实训报告总结收获.doc
  10. 三级医院信息互联互通标准化成熟度测评要求