【初识数据结构】CS61B中的最小生成树问题

本教程总结CS61B 关于图章节中的最小生成树(Minimum Spanning Trees, MST)问题,以及对应的的算法

什么是最小生成树(MST)

考虑这样一个问题,给你一个无向图,你能不能找出这个图中的一组边,让其满足:

  • 连通的
  • 无环的
  • 涉及到了所有的结点
    比如这样一张图
    alt text
    这样的一组边构成的树,称为生成树
    而让这个树中所有的边权重之和最小即为最小生成树

如何找到最小生成树

一个非常有用的性质:割边属性(Cut Property)

  • 一个 cut 就是将一个图中的结点分成两个非空集合
  • 一个 crossing cut 就是从一个集合结点连接到了另一个集合结点的边
    割边属性是这样描述的:给定任意一个 cut,最小权重的 crossing edge 一定在最小生成树中
    alt text
    比如上面这张图中,所有的红色边都是 crossing edge,其中最小的边一定在最小生成树中
    我们判断 crossing edge 只需要看这个边是否连接了两个属于不同集合(也就是不同颜色)的结点即可

通用的寻找 MST 算法

基于割边属性,我们可以这样构造一个算法:
首先一开始MST没有边:

  • 找到一个没有 crossing edge 在MST中的 cut
  • 然后将最小权重的 crossing edge 加入 MST
  • 一直重复到 V-1 条边

Prim’s Algorithm

相比较最短路径树而言,最小生成树并不需要给出一个起点,在最小生成树中,只需要给你图然后说告诉我哪些边触及所有顶点且权重之和最小即可。

但是并不代表我们不选取一个起始结点,相对来说,Prim 算法会随机选择一个起始结点,这个随机性并不对最终结果产生影响(我们总要寻找一个结点来入手)

算法流程

从一个随机结点开始:

  • 将一个一头连接已经在MST中结点的最短的边,加入MST,直到 V-1 条边

alt text
比如在这个图中,所有符合上述条件的即为红色的边:连接了一个在MST中的结点
然后我们选取这些边中最短的边,也就是A->B或者E->D

Prim 算法实际上是不断地使用了割边属性

改进的 Prim’s Algorithm

Prim 算法是可以奏效的,但效率太低了,因为你必须考虑大量的穿过这个 cut 的 crossing edge
我们对 Prim 算法进行改进:

  • 将所有结点放入 fringe PQ 中,以结点到树的距离作为排序标准
  • 重复:将距离树最近的结点移出,然后将其指出的所有边进行relaxation,然后对 distTo 和 edgeTo 数组进行更新

alt text
这个图中我们刚刚把距离树最近的结点E移出,加入到MST中,然后relax它的所有的边,同时更新 distTo 和 edgeTo 数组

图中加粗的边为MST中的边,图中的虚线表示relax出来的边,作为MST的候选边,如果在relax中发现这条边不如之前的边(比如C->B),或者这条边被后续的边所取代(比如下一步的C->F即将被E->F取代),那么这条边我们甚至不标注为虚线

而如果在MST中的边被新边取代,那么我们把新边加入MST,原来的边回到最开始的模样(不加粗,也不标为虚线)

同时我们看到 Fringe 中的结点是按照到树的距离进行排序的,同时 distTo 数组也变为了到树的距离

算法实现

public PrimMST{public PrimMST(EdgeWeightedGraph G){edgeTo = new Edge[G.V()];distTo = new double[G.V()];marked = new boolean[G.V()];fringe = new SpecialPQ<Double>[G.V()];distTo[s] = 0.0;setDDistancesToInfinityExceptS(s);insertAllVertices(fringe);while(!fringe.isEmpty()){int v = fringe.delMin();scan(G,v);}} ....private scan(EdgeWeightedGraph G, int v){marked[v] = true;for(Edge e: G.adj(v)){int w = e.other(v);if(marked[w]){continue;}if(e.weight() < distTo[w]){distTo[w] = e.weight();edgeTo[w] = e;pq.decreasePriority(w, distTo[w]);}}}
}

Kruskal’s Algorithm

按照权重顺序考虑所有的边,只要这条边加入MST后不构成环,那么就将它加入MST中,重复直到 V-1 条边

Kruskal 算法计算过程中创造出的MST可能是割裂不连续的,但是没关系,最后会得出正确的结果

我们可以通过维护两个辅助数据结构来帮助我们实现:

  • WQU 加权快速集合:帮助我们判断是否有环生成
  • MST :统计最小生成树的边

alt text
在这个图中我们按顺序从 Fringe 中取出对应的边,当我们即将取出E->B这条边时,WQU告诉我们已经有了一条从E到B的路径,也就是下一步即将成环,所以我们不考虑E->B这条边

算法实现

public class KruskalMST{private List<Edge> mst = new ArrayList<Edge>();public KruskalMST(EdgeWeighedGraph G){MinPQ<Edge> pq = new MinPQ<Edge>();for(Edge e: G.edges()){pq.insert(e);}WeighedQuickUnionPC uf = new WeighedQuickUnionPC(G.V());while(!pq.isEmpty()){Edge e = pq.delMin();int v = e.from();int w = e.to();if(!uf.connected(v,w)){uf.union(v,w);mst.add(e);}}}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/news/915702.shtml
繁体地址,请注明出处:http://hk.pswp.cn/news/915702.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

vue apk返回键不好使

在 Android 设备上&#xff0c;你可以通过监听物理返回键来实现特定的逻辑。这可以通过在 Vue 组件中添加一个事件监听器来实现&#xff1a;mounted() {this.$once(hook:beforeDestroy, () > {if (document.removeEventListener) {document.removeEventListener(backbutton,…

Ubuntu 22.04 安装 MySQL 8.0 完整步骤文档

1、安装 1.1、下载 cd /usr/local/在 /usr/local/ 下执行&#xff0c;下载资源包&#xff0c;可以本地下载上传 wget https://downloads.mysql.com/archives/get/p/23/file/mysql-8.0.32-linux-glibc2.12-x86_64.tar.xz1.2、解压安装 tar -Jxvf mysql-8.0.32-linux-glibc2.…

Docker,其他机器下载镜像并copy到目标机器导入docker镜像

Docker&#xff0c;其他机器下载镜像并copy到目标机器导入docker镜像源机器 【下载镜像】目标机器slave1 【无法下载镜像】步骤 1&#xff1a;在网络正常的机器&#xff08;cg&#xff09;上下载镜像&#xff0c;导出镜像到指定路径# 1. 下载镜像docker pull ubuntu:20.04# 2.…

基于现代R语言【Tidyverse、Tidymodel】的机器学习方法与案例分析

机器学习已经成为继理论、实验和数值计算之后的科研“第四范式”&#xff0c;是发现新规律&#xff0c;总结和分析实验结果的利器。机器学习涉及的理论和方法繁多&#xff0c;编程相当复杂&#xff0c;一直是阻碍机器学习大范围应用的主要困难之一&#xff0c;由此诞生了Python…

如何将 git 远程 URL 从 https 更改为 ssh

在项目开发中&#xff0c;使用 SSH 连接 Git 仓库可以提高安全性和便利性。本文将指导你如何将 Git 远程 URL 从 HTTPS 更改为 SSH。操作指南步骤 1: 查看当前远程 URL首先&#xff0c;确认当前的远程 URL 使用的是 https。打开终端并输入以下命令&#xff1a;git remote -v如&…

PyCharm 高效入门指南(核心模块详解二)

四、生产力工具集成PyCharm 不仅仅是 Python 编辑器&#xff0c;更是集成了多种开发工具的综合平台。通过内置的生产力工具&#xff0c;开发者可以在一个界面内完成数据库操作、科学计算、远程开发和测试等全流程工作&#xff0c;避免工具切换带来的效率损耗。4.1 数据库工具链…

WebkitSpeechRecognition 语音识别

JavaScript WebkitSpeechRecognition:使用语音识别技术增强 Web 应用程序 WebkitSpeechRecognition 是一种 JavaScript API,它可以让您的 Web 应用程序使用语音识别技术。使用 WebkitSpeechRecognition,您可以让用户通过说话来与您的 Web 应用程序进行交互,这可以使您的应…

CUDA C++核心库(CCCL)

文章目录CUDA C核心库&#xff08;CCCL&#xff09;核心库介绍CUDA C 开发工具的层级范围各层级工具的具体内容Thrust自动内存管理类型安全自定义分配器&#xff08;页锁定内存&#xff09;高级API替代底层操作thrust::transform基本使用几种执行策略iteratorload_cs高效索引md…

MySQL InnoDB存储引擎深度解析:从原理到优化

InnoDB的优势InnoDB之所以成为众多应用的首选&#xff0c;主要得益于以下几个显著优势&#xff1a;事务支持&#xff1a;InnoDB是MySQL中唯一支持ACID&#xff08;原子性、一致性、隔离性、持久性&#xff09;事务的存储引擎。它通过日志和锁机制确保事务的完整性&#xff0c;这…

LLM评测框架Ragas:Natural Language Comparison指标(解决了Ollama推理框架不支持的问题)

Factural Correctness Factural Correctness是事实正确性是评价LLM生成的反馈和reference的事实正确性。该指标用于确定生成的响应与参考文献的一致程度。Factural Correctness取值在0到1之间,越接近于1结果越好。 为了衡量回应和参考文献之间的一致性,该指标使用 LLM 首先将…

HTTP 协议常见字段(请求头/响应头)

HTTP&#xff08;HyperText Transfer Protocol&#xff09;协议通过 请求头&#xff08;Request Headers&#xff09; 和 响应头&#xff08;Response Headers&#xff09; 传递元数据。以下是 最常见的 HTTP 字段 及其作用&#xff1a;1. 通用字段&#xff08;请求和响应均可使…

期货配资软件开发注意事项?

期货配资软件开发 期货配资软件开发涉及多个核心模块&#xff0c;包括资金管理、风险控制、交易接口、用户权限管理等。此类系统需符合金融监管要求&#xff0c;确保资金安全与数据合规。开发过程中需优先考虑高并发、低延迟及系统稳定性。期货资管系统平台搭建方案架构设计 采…

STM32-第十节-DMA直接存储器存取

一、DMA&#xff1a;1.简介&#xff1a;DMA&#xff0c;直接存储区存取DMA可以提供外设和存储器或存储器与存储器见的高速数据传输&#xff0c;无需CPU干预。12个通道&#xff1a;DMA1&#xff08;7个通道&#xff09;&#xff0c;DMA2&#xff08;5个通道&#xff09;每个通道…

服务器设置国外IP无法访问对防御攻击有用吗?

将服务器设置为仅允许国外 IP 访问&#xff0c;限制国内 IP 访问&#xff0c;确实可以在某些特定场景下提高服务器的抗攻击能力&#xff0c;但这并不能完全防御攻击。以下是对这种方法的分析、优缺点以及其他防御攻击的补充措施。1. 仅允许国外 IP 访问是否有用&#xff1f;1.1…

八大作业票(一) 动火安全作业证

动火安全作业证 执行标准:GB30871 GSDH——2200001 申报单位 申请人 作业申请时间 年 月 日 时 分 动火内容 动火方式 动火地点 动火类别 特级动火□ 一级动火□ 二级动火□ 作业负责人 监护人 动火…

NumPy库使用教学,简单详细。

NumPy 使用教学NumPy 是 Python 中用于科学计算的基础库&#xff0c;它提供了高性能的多维数组对象以及用于处理这些数组的工具。下面将结合多个代码文件&#xff0c;详细介绍 NumPy 的各种用法。1. 创建数组1.1 从列表创建数组import numpy as np# 一维数组 list1 [1,2,3,4,5…

vue3:十八、内容管理-实现行内图片的预览、审核功能

一、实现效果 实现图片的显示,大图预览;审核部分的待审核的审核功能 二、图片预览实现 1、参考官网 官网-图片预览 2、图片预览插槽设置 {row,index} 插槽中获取row行信息、index索引信息(指定行图片预览需要用到) style 设置基本样式宽width高height src 设置图片的路径…

Go后端配置文件教程

注&#xff1a;本文为博主&#xff0c;首次接触项目时的入门级配置实操在 Go 后端中&#xff0c;使用配置文件管理参数&#xff08;如数据库连接、服务端口等&#xff09;是必备技能。Viper 是 Go 生态中最流行的配置管理库。支持多种配置文件、环境变量、命令行参数等&#xf…

ubuntu24.04安装CUDA、VLLM、Pytorch等并部署Qwen3-8B-AWQ【50系显卡通用】

1. 系统更新与依赖安装 sudo apt update && sudo apt upgrade -y sudo apt install -y python3-pip python3-venv build-essential git nvidia-driver-575注:RTX 5070 Ti 推荐驱动 ≥550 版本 我是直接官网安装最新的驱动了,反正向上兼容,驱动安装教程可以参考我以…

Azure可靠性架构指南:构建云时代的高可用系统

随着企业加速拥抱数字化转型&#xff0c;云服务的可靠性已成为业务连续性的核心命题。Microsoft Azure凭借其"可靠性即核心"的设计理念&#xff0c;为企业技术决策者与架构师提供了一个可信赖的数字化底座。本文将系统解析Azure如何通过技术架构、工具链与方法论&…