【开卷数据结构 】图的应用——最短生成树

简介: 【开卷数据结构 】图的应用——最短生成树

最小生成树


Q:什么是广度优先搜索


A:一个连通图的生成树含有图中全部的顶点,并且只含尽可能少的边。若砍去它的一条边,则会使生成树变成非连通图。若给它增加一条边,则会形成图中的一条回路。


对于一个带权连通无向图 G=(V, E) ,生成树不同,每棵树的权(即树中所有边上的权值之和)也可能不同,其中边的权值之和最小的那棵生成树(构造连通网的最小代价生成树),称为G的最小生成树。


通过定义不难看出,最小生成树具有以下性质:


1)最小生成树不是唯一的,即最小生成树的树形不唯一,所有生成树中可能有多个最小生成树。当图 G 中的各边权值互不相等时,G 的最小生成树是唯一的。 若无向连通图 G 的边数比顶点数少1,即 G 本身是一棵树时,则 G 的最小生成树就是它本身。

2)最小生成树的边的权值之和总是唯一的,虽然最小生成树不唯一,但其对应的边的权值之和总是唯一的,而且是最小的。

3)最小生成树的边数为顶点数减1。

构造最小生成树 有多种算法,但大多数算法都利用了最小生成树的下列性质:假设  G=(V,E) 是一个带权连通无向图,U 是顶点集 V 的一个非空子集。若 (u,v) 是一条具有最小权值的边,其中  u∈U,v∈V−U , 则必存在一棵包含边 (u,v) 的最小生成树。


下面介绍一个通用的最小生成树算法:


GENERIC_MST(G){
  T=NULL;
  while T 未形成一棵生成树;
  do 找到一条最小代价边(u, v)并且加入T后不会产生回路;
    T=T ∪ (u, v);
}

通用算法每次加入一条边以逐渐形成一棵生成树,下面介绍两种实现上述通用算法的途径。


Prim算法实现最小生成树


Q:如何通过Prim算法构建最小生成树


A:初始时从图中任取一顶点加入树 T,此时树中只含有一个顶点。之后选择一个与当前 T 中顶点集合距离最近的顶点,并将该顶点和相应的边加入 T。每次操作后 T 中的顶点数和边数都增 1。以此类推,直至图中所有的顶点都并入 T,得到的 T 就是最小生成树。此时 T 中必然有 n-1条边。


通俗来讲,Prim算法构建最小生成树流程如下:


1) 从某顶点 u0 出发,选择与它关联的具有最小权值的边 (u0, v),将其顶点 v 加入到生成树的顶点集合 U 中。

2) 每一步从一个顶点在 U 中,而另一个顶点不在 U 中的各条边中选择权值最小的边 (u, v),把它的顶点 v 加入到 U 中。

3)直到所有顶点都加入到生成树顶点集合 U 中为止。

Prim算法的简单实现如下:


void Prim(G,T)
{
  t=NULL;    //初始化空树 
  U={w};    //添加任一结点 w
  while((V-U)!=NULL)  //若树中不含全部结点
  {
  设(u,v)是使 u∈U与v∈(V-U),且权值最小的边;
  T=T∪{{u,v}} //边归入树 
  U=U∪{v};  //顶点归入树 
  } 
}

Prim 算法的时间复杂度为 O(n^2),不依赖于|E|,因此它适用于求解边稠密的图的最小生成树。


kruskal算法实现最小生成树


与 Prim 算法从顶点开始扩展最小生成树不同,Kruskal 算法是一种按权值的递增次序选择合适的边来构造最小生成树的方法。


Q:如何通过kruskal算法构建最小生成树


A:初始时为只有 n 个顶点而无边的非连通图 T=V,每个顶点自成一个连通分量,然后按照边的权值由小到大的顺序,不断选取当前未被选取过且权值最小的边,若该边依附的顶点落在 T 中不同的连通分量上,则将此边加入 T,否则舍弃此边而选择下一条权值最小的边。以此类推,直至 T 中所有顶点都在一个连通分量上。


通俗来讲,kruskal算法构建最小生成树流程如下:


1)设连通网络 N = { V, E },构造一个只有 n 个顶点,没有边的非连通图 T = { V, Ø }, 每个顶点自成一个连通分量。

2) 在 E 中选最小权值的边,若该边的两个顶点落在不同的连通分量上,则加入 T 中;否则舍去,重新选择。

3)重复下去,直到所有顶点在同一连通分量上为止。

kruskal算法的简单实现如下::


void Kruskal(V,T)
{
  T=V;    //初始化树T,仅含顶点
  numS=n    //连通分量数
  while(numS>1){  //若连通分量数大于 1 
  从 E 中取出权值最小的边(v,u); 
  T=T ∪{{v,u}}; //将此边加入生成树中 
  numS--;   //连通分量减 1 
  } 
}

kruskal 算法的时间复杂度为 O(ElogE) ,因此它适用于求解边稀疏而顶点多的图的最小生成树。


相关文章
|
4月前
|
消息中间件 缓存 NoSQL
Redis各类数据结构详细介绍及其在Go语言Gin框架下实践应用
这只是利用Go语言和Gin框架与Redis交互最基础部分展示;根据具体业务需求可能需要更复杂查询、事务处理或订阅发布功能实现更多高级特性应用场景。
331 86
|
6月前
|
存储 监控 安全
企业上网监控系统中红黑树数据结构的 Python 算法实现与应用研究
企业上网监控系统需高效处理海量数据,传统数据结构存在性能瓶颈。红黑树通过自平衡机制,确保查找、插入、删除操作的时间复杂度稳定在 O(log n),适用于网络记录存储、设备信息维护及安全事件排序等场景。本文分析红黑树的理论基础、应用场景及 Python 实现,并探讨其在企业监控系统中的实践价值,提升系统性能与稳定性。
192 1
|
6月前
|
存储 监控 算法
公司员工泄密防护体系中跳表数据结构及其 Go 语言算法的应用研究
在数字化办公中,企业面临员工泄密风险。本文探讨使用跳表(Skip List)数据结构优化泄密防护系统,提升敏感数据监测效率。跳表以其高效的动态数据处理能力,为企业信息安全管理提供了可靠技术支持。
156 0
|
10月前
|
算法 Java
算法系列之数据结构-Huffman树
Huffman树(哈夫曼树)又称最优二叉树,是一种带权路径长度最短的二叉树,常用于信息传输、数据压缩等方面。它的构造基于字符出现的频率,通过将频率较低的字符组合在一起,最终形成一棵树。在Huffman树中,每个叶节点代表一个字符,而每个字符的编码则是从根节点到叶节点的路径所对应的二进制序列。
328 3
 算法系列之数据结构-Huffman树
|
10月前
|
存储 自然语言处理 数据库
【数据结构进阶】AVL树深度剖析 + 实现(附源码)
在深入探讨了AVL树的原理和实现后,我们不难发现,这种数据结构不仅优雅地解决了传统二叉搜索树可能面临的性能退化问题,还通过其独特的平衡机制,确保了在任何情况下都能提供稳定且高效的查找、插入和删除操作。
804 19
|
存储 C++
【C++数据结构——树】哈夫曼树(头歌实践教学平台习题) 【合集】
【数据结构——树】哈夫曼树(头歌实践教学平台习题)【合集】目录 任务描述 相关知识 测试说明 我的通关代码: 测试结果:任务描述 本关任务:编写一个程序构建哈夫曼树和生成哈夫曼编码。 相关知识 为了完成本关任务,你需要掌握: 1.如何构建哈夫曼树, 2.如何生成哈夫曼编码。 测试说明 平台会对你编写的代码进行测试: 测试输入: 1192677541518462450242195190181174157138124123 (用户分别输入所列单词的频度) 预
477 14
【C++数据结构——树】哈夫曼树(头歌实践教学平台习题) 【合集】
|
11月前
|
DataX
☀☀☀☀☀☀☀有关栈和队列应用的oj题讲解☼☼☼☼☼☼☼
### 简介 本文介绍了三种数据结构的实现方法:用两个队列实现栈、用两个栈实现队列以及设计循环队列。具体思路如下: 1. **用两个队列实现栈**: - 插入元素时,选择非空队列进行插入。 - 移除栈顶元素时,将非空队列中的元素依次转移到另一个队列,直到只剩下一个元素,然后弹出该元素。 - 判空条件为两个队列均为空。 2. **用两个栈实现队列**: - 插入元素时,选择非空栈进行插入。 - 移除队首元素时,将非空栈中的元素依次转移到另一个栈,再将这些元素重新放回原栈以保持顺序。 - 判空条件为两个栈均为空。
|
Java C++
【C++数据结构——树】二叉树的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现二叉树的基本运算。​ 相关知识 创建二叉树 销毁二叉树 查找结点 求二叉树的高度 输出二叉树 //二叉树节点结构体定义 structTreeNode{ intval; TreeNode*left; TreeNode*right; TreeNode(intx):val(x),left(NULL),right(NULL){} }; 创建二叉树 //创建二叉树函数(简单示例,手动构建) TreeNode*create
388 12
|
C++
【C++数据结构——树】二叉树的性质(头歌实践教学平台习题)【合集】
本文档介绍了如何根据二叉树的括号表示串创建二叉树,并计算其结点个数、叶子结点个数、某结点的层次和二叉树的宽度。主要内容包括: 1. **定义二叉树节点结构体**:定义了包含节点值、左子节点指针和右子节点指针的结构体。 2. **实现构建二叉树的函数**:通过解析括号表示串,递归地构建二叉树的各个节点及其子树。 3. **使用示例**:展示了如何调用 `buildTree` 函数构建二叉树并进行简单验证。 4. **计算二叉树属性**: - 计算二叉树节点个数。 - 计算二叉树叶子节点个数。 - 计算某节点的层次。 - 计算二叉树的宽度。 最后,提供了测试说明及通关代
206 10
|
存储 算法 测试技术
【C++数据结构——树】二叉树的遍历算法(头歌教学实验平台习题) 【合集】
本任务旨在实现二叉树的遍历,包括先序、中序、后序和层次遍历。首先介绍了二叉树的基本概念与结构定义,并通过C++代码示例展示了如何定义二叉树节点及构建二叉树。接着详细讲解了四种遍历方法的递归实现逻辑,以及层次遍历中队列的应用。最后提供了测试用例和预期输出,确保代码正确性。通过这些内容,帮助读者理解并掌握二叉树遍历的核心思想与实现技巧。
581 3