蓝桥杯备考:图论之Prim算法

embedded/2025/3/21 3:27:03/

嗯。通过我们前面的学习,我们知道了,一个具有n个顶点的连通图,它的生成树包括n-1个边,如果边多一条就会变成图,少一条就不连通了

接下来我们来学一下把图变成生成树的一个算法

Prim算法,我们从任意一个结点开始构造最小生成树,先把离生成树最近的点拉进来,然后更新所有结点距离生成树的距离,然后再拉进来最近的点,然后再更新所有结点到这颗生成树的距离,一直到所有结点全部拉进来的时候,我们的算法就结束了

这里我们需要开两个数组,一个是dist数组来判断某个点距离生成树的距离,另一个是st数组,来判断某个点在不在生成树里

如图,我们把起始点加入进去了,接下来我们更新一下所有点离最小生成树的距离

我们还是找距离生成树最近的结点,如图,是5,我们把5拉进去

接下来把5标记上,然后更新所有点距离最小生成树的距离

然后继续找最小的,应该是2结点了

把2结点拉进去之后,我们再次更新dist数组,....如此往复,直到所有点都遍历完之后,我们的最小生成树也就构建完了;

话不多说我们来实现一下代码

我们先来实现一下邻接矩阵的代码;

这道题我们还要考虑一下连通不连通的情况

比如这张图,当我们用prim算法把2,1,5,4都加到生成树里的时候,3这个结点距离生成树的值还是无穷,就说明他不连通,没法找最小生成树

下面就是我们的代码

#include <iostream>
#include <cstring>
using namespace std;
const int N = 5010;
const int INF = 0x3f3f3f3f;
int dist[N];
bool st[N];
int edges[N][N];
int n, m;
int ret = 0;
int prim()
{dist[1] = 0;for (int i = 1; i <= n; i++)//循环把n个结点加入到生成树里面 {//1.找最近点int t = 0;for (int j = 1; j <= n; j++){if (dist[j] < dist[t] && !st[j]){t = j;}}if (dist[t] == INF) return INF;//不连通//把最小结点加入到生成树里st[t] = true;ret += dist[t];//更新所有结点离最小生成树的距离for (int i = 1; i <= n; i++){dist[i] = min(dist[i], edges[t][i]);}}return ret;
}
int main()
{cin >> n >> m;memset(dist, 0x3f, sizeof(dist));memset(edges, 0x3f, sizeof(edges));for (int i = 1; i <= m; i++){int a, b, c; cin >> a >> b >> c;edges[a][b] = edges[b][a] = min(edges[a][b],c);}ret = prim();if (ret == INF) cout << "orz" << endl;else cout << ret << endl;return 0;
}

接下来,我们用邻接表实现一下这道题

#include <iostream>
#include <cstring>
#include <vector>
using namespace std;
typedef pair<int, int> PII;
const int N = 5010, INF = 0x3f3f3f3f;
vector <PII> path[N];
int dist[N];
bool st[N];
int ret = 0;
int n, m;
int prim()
{dist[1] = 0;for (int i = 1; i <= n; i++){int t = 0;for (int j = 1; j <= n; j++){if (dist[t] > dist[j] && !st[j]){t = j;}}if (dist[t] == INF) return INF;st[t] = true;ret += dist[t];for (auto &e : path[t]){int x = e.first, y = e.second;dist[x] = min(dist[x], y);}}return ret;
}
int main()
{memset(dist, 0x3f, sizeof(dist));cin >> n >> m;for (int i = 1; i <= m; i++){int x, y, z; cin >> x >> y >> z;path[x].push_back({ y,z });path[y].push_back({ x,z });}int r = prim();if (r == INF) cout << "orz" << endl;else cout << r << endl;return 0;
}

根据我们的代码可以看到,我们的prim算法时间复杂度是N²,如果结点很少边很多的话,我们用prim算法最合适


http://www.ppmy.cn/embedded/174315.html

相关文章

远程控制中的云电脑是什么意思?1分钟学会用

很多常用我们ToDesk远程控制的朋友们或许会注意到无论是在PC端还是移动端中都出现有【云电脑】【来云电脑爽玩-新用户免费1小时】这些词句等信息。那么这究竟是代表什么意思呐&#xff1f;云电脑是什么又怎么用呐&#xff1f;为什么要增加云电脑&#xff1f;以下小编就为大家科…

OpenHarmony子系统开发 - 电池管理(一)

OpenHarmony子系统开发 - 电池管理&#xff08;一&#xff09; 一、电量与LED灯颜色的定制开发指导 概述 简介 OpenHarmony默认提供了电量与LED灯颜色的映射关系。对于部分产品形态&#xff08;如Pad&#xff09;&#xff0c;会使用LED灯的颜色来展示当前设备充电时的电量信…

物联网为什么用MQTT不用 HTTP 或 UDP?

先来两个代码对比&#xff0c;上传温度数据给服务器。 MQTT代码示例 // MQTT 客户端连接到 MQTT 服务器 mqttClient.connect("mqtt://broker.server.com:8883", clientId) // 订阅特定主题 mqttClient.subscribe("sensor/data", qos1) // …

1.Qt SDK 的下载和安装

1Qt 下载官⽹&#xff1a; http://download.qt.io/archive/qt/ 2版本自行选择 3下载对应版本的.exe文件 4下载包下载完成 5双击.exe文件&#xff0c;默认下一步&#xff0c;要注册一个qt的账户 6记住程序安装的位置&#xff0c;后面要配置环境变量 7勾3个&#xff08;组件自行…

Spring Boot 整合 Nacos 注册中心终极指南

摘要&#xff1a;本文详细讲解如何在 Spring Boot 项目中整合 Nacos 作为服务注册中心&#xff0c;包含版本选择、核心配置、心跳机制及常见问题解决方案&#xff0c;助你快速构建微服务架构。 一、环境准备 1.1 组件版本要求 组件推荐版本说明Spring Boot2.6.x长期支持版本S…

Unity音乐内存优化

文章目录 音乐下载远程音乐 音乐 音乐文件如果只从工程目录里面读取&#xff0c;那有很多种方法可以优化&#xff0c;比如设置Load Type直接采用流式加载方式&#xff0c;内存直接降最小&#xff08;但是记住&#xff0c;每种优化都是有对应的代价的&#xff0c;优化是一种平衡…

【八股文】ArrayList和LinkedList的区别

先讲讲两者是如何实现的 ArrayList public class ArrayList<E> extends AbstractList<E>implements List<E>, RandomAccess, Cloneable, java.io.Serializable {transient Object[] elementData; private int size; } 通过源码可以看出&#xff0c;ArrayLis…

Android Audio基础(18)——最小缓冲区

在创建 AudioTrack 时有一个缓冲区大小的参数&#xff0c;最小缓冲区参数通过 AudioTrack.getMinBufferSize() 获取。 一、最小缓冲区 为了让音频数据通路能正常运转&#xff0c;共享FIFO必须达到最小缓冲区的大小。如果数据缓冲区分配得过小&#xff0c;那么播放声音会频繁遭…