DSA 经典数据结构与算法 学习心得和知识总结(三) |有向无环图及其应用

本文主要是介绍DSA 经典数据结构与算法 学习心得和知识总结(三) |有向无环图及其应用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!


注:提前言明 本文借鉴了以下博主、书籍或网站的内容,其列表如下:

1、参考书籍:《算法导论》第三版      就是这本被封神的杰作,就是它🤦
2、参考书籍:《数据结构》严奶奶版
3、参考书籍:《数据结构》(用面向对象方法与C++语言描述) 第二版 殷人昆版
4、参考书籍:《数据结构》(C++版) 第三版 邓俊辉版
5、华中科技大学 有向无环图及应用 公开课,点击前往
6、OI Wiki 有向无环图,点击前往
7、OI Wiki 拓扑排序,点击前往
8、关键路径 + 拓扑排序,点击前往


DSA 经典数据结构与算法 有向无环图

  • 文章快速说明索引
  • 有向无环图的背景
    • 拓扑排序
    • 逆拓扑排序
    • AOV 网
    • 关键路径和 AOE 网


在这里插入图片描述


文章快速说明索引

学习目标:

前言:还记得在大学的时候,数据结构作为计算机科学与技术专业最重要的一门课 当时学校采用的教材是严奶奶的粉红色那本,不过当时是真的不愿多看一眼 😂 苦涩难懂 又非常深奥,满篇伪代码实现的例子和夏日那十分惬意的下午 简直让人头大而晕!

也可能是上学那会儿年少浮躁,也可能是因为当时的能力比较的菜吧 ┑( ̄Д  ̄)┍ 。现在再回头捧读厚厚的《算法导论》,竟然有一种说不上来的快乐 沉浸在数据结构和算法之美,惊叹于高超技巧式拍案惊奇!


学习内容:(详见目录)

1、数据结构与算法(DSA)之有向无环图


学习时间:

2024年02月15日 14:17:05


学习产出:

1、CSDN 技术博客 1篇


有向无环图的背景

有向无环图(Directed Acyclic Graph):一个无环的有向图。

  1. 其性质如下:
  • 拓扑排序 的图,一定是有向无环图;如果有环,那么环上的任意两个节点在任意序列中都不满足条件了
  • 有向无环图,一定能拓扑排序;(归纳法)假设节点数不超过 k 的 有向无环图都能拓扑排序,那么对于节点数等于 k 的,考虑执行拓扑排序第一步之后的情形即可

  1. 如何判定一个图是否是有向无环图呢?
  • 检验它是否可以进行 拓扑排序 即可
  • 当然也有另外的方法,可以对图进行一遍 DFS,在得到的 DFS 树上看看有没有连向祖先的非树边(返祖边)。如果有的话,那就有环了

接下来看一下 DAG 的应用, 如下:

一、描述表达式:

// 如下表达式含 + * 操作((a + b) * (b * (c + d)) + (c + d) * e) * ((c + d) * e)

用二叉树表示这个表达式,(21个顶点)如下:

在这里插入图片描述

用有向无环图表示该表达式,(12个顶点)如下:

在这里插入图片描述


二、表示 AOV网 (Activity On Vertex Network) or AOE网(Activity On Edge)

在这里插入图片描述

  • 什么是AOV网与AOE网?——以及AOV网与AOE网区别和运用,点击前往

下面我们再详细介绍它们!


拓扑排序

拓扑排序的英文名是 Topological sorting。拓扑排序要解决的问题是给一个有向无环图的所有节点排序。换言之:其是一个有向无环图(DAG,Directed Acyclic Graph)的所有顶点的线性序列,且该序列必须满足下面两个条件:

  • 每个顶点出现且只出现一次
  • 若存在一条从顶点 A 到顶点 B 的路径,那么在序列中顶点 A 出现在顶点 B 的前面
  • 有向无环图(DAG)才有拓扑排序,非DAG就没有拓扑排序一说

一个经典的案例,如下:

在这里插入图片描述

因此我们可以说:

  • 在一个 DAG(有向无环图) 中,我们将图中的顶点以线性方式进行排序,使得对于任何的顶点 u 到 v 的有向边 (u,v),都可以有 u 在 v 的前面
  • 还有给定一个 DAG,如果从 i 到 j 有边,则认为 j 依赖于 i。如果 i 到 j 有路径(i 可达 j),则称 j 间接依赖于 i
  • 拓扑排序的目标是将所有节点排序,使得排在前面的节点不能依赖于排在后面的节点

举一个例子,如下:

  • 拓扑排序(Topological Sorting),点击前往

在这里插入图片描述


逆拓扑排序

逆拓扑排序的步骤:

  • 从AOV网中选择一个出度为0的顶点并输出
  • 从网中删除该顶点和所有以它为终点的有向边
  • 重复1和2,直到当前的AOV网为空

AOV 网

日常生活中,一项大的工程可以看作是由若干个子工程组成的集合,这些子工程之间必定存在一定的先后顺序,即某些子工程必须在其他的一些子工程完成后才能开始。

我们用有向图来表现子工程之间的先后关系,子工程之间的先后关系为有向边,这种有向图称为顶点活动网络,即 AOV 网 (Activity On Vertex Network)。一个 AOV 网必定是一个有向无环图,即不带有回路。与 DAG 不同的是,AOV 的活动都表示在边上。

在 AOV 网中,顶点表示活动,弧表示活动间的优先关系。AOV 网中不应该出现环,这样就能够找到一个顶点序列,使得每个顶点代表的活动的前驱活动都排在该顶点的前面,这样的序列称为拓扑序列(一个 AOV 网的拓扑序列不是唯一的),由 AOV 网构造拓扑序列的过程称为拓扑排序。因此,拓扑排序也可以解释为将 AOV 网中所有活动排成一个序列,使得每个活动的前驱活动都排在该活动的前面(一个 AOV 网中的拓扑排序也不是唯一的)。

  • 前驱活动:有向边起点的活动称为终点的前驱活动(只有当一个活动的前驱全部都完成后,这个活动才能进行)
  • 后继活动:有向边终点的活动称为起点的后继活动

检测 AOV 网中是否带环的方式是构造拓扑序列,看是否包含所有顶点。构造这个拓扑序列步骤:

  1. 从图中选择一个入度为零的点
  2. 输出该顶点,从图中删除此顶点及其所有的出边
  3. 重复上面两步,直到所有顶点都输出,拓扑排序完成,或者图中不存在入度为零的点,此时说明图是有环图,拓扑排序无法完成,陷入死锁

关键路径和 AOE 网

与 AOV 网对应的是 AOE 网(Activity On Edge Network) 即边表示活动的网。AOE 网是一个带权的有向无环图,其中,顶点表示事件,弧表示活动持续的时间。通常,AOE 网可以用来估算工程的完成时间。AOE 网应该是无环的,且存在唯一入度为零的起始顶点(源点),以及唯一出度为零的完成顶点(汇点)。

在这里插入图片描述

AOE 网中的有些活动是可以并行进行的,所以完成整个工程的最短时间是从开始点到完成点的最长活动路径长度(这里所说的路径长度是指路径上各活动的持续时间之和,即弧的权值之和,不是路径上弧的数目)。因为一项工程需要完成所有工程内的活动,所以最长的活动路径也是关键路径,它决定工程完成的总时间。


AOE 网的相关基本概念,如下:

  • 活动:AOE 网中,弧表示活动。弧的权值表示活动持续的时间,活动在事件被触发后开始。
  • 事件:AOE 网中,顶点表示事件,事件能被触发。

  • 弧(活动)aj 的最早开始时间:初始点到该弧起点的最长路径长度,记为 e(j)。
  • 弧(活动)aj 的最迟开始时间:在不推迟整个工期的前提下,工程达到弧起点所表示的状态最晚能容忍的时间,记为 l(j)。即:事件的最迟发生时间 - 弧的活动时间值。

  • 顶点(事件)vj 的最早发生时间:初始点到该顶点的最长路径长度,记为 ve(j),它决定了以该顶点开始的活动的最早发生时间,所以 ve(j) = e(j)。
  • 顶点(事件)vj 的最迟发生时间:在不推迟整个工期的前提下,工程达到顶点所表示的状态最晚能容忍的时间,记为 vl(j),它决定了所有以该状态结束的活动的最迟发生时间,所以 l(j) = vl(j) - dul(aj)。

  • 关键路径:AOE 网中从源点到汇点的最长路径的长度。
  • 关键活动:关键路径上的活动,最早开始时间和最迟开始时间相等(看下面时间余量d(j) = 0的)。

最早和最迟发生时间的递推关系:

在这里插入图片描述

按拓扑顺序求,最早是从前往后,前驱顶点的最早开始时间与边的权重之和最大者;最迟是从后往前,后继顶点的最迟开始时间与边的权重之差的最小者。


下面看一个例子,计算如下:

在这里插入图片描述

如上图,其其中之一的拓扑排序,如下:

V1 V3 V2 V5 V4 V6
V1V2V3V4V5V6
ve(j):事件 的最早发生时间 ->0326 max68 max
vl(j):事件 的最迟发生时间 <-04 min2 min678
a1a2a3a4a5a6a7a8
e(j):活动 aj 的最早开始时间 ->00332266
l(j):活动 aj 的最迟开始时间 <-4 - 32 - 26 - 27 - 36 - 48 - 38 - 28 - 1
l(j):活动 aj 的最迟开始时间 <-10442567
d(j):活动 aj 的时间余量 l(j) - e(j)10110301

于是关键活动有:a2 a5 a7。如下:

在这里插入图片描述

V1->V3->V4->V6 就是关键路径,total = 8!


关键路径算法:

  1. 输入 e 条弧 (j,k),建立 AOE 网;
  2. 从源点 v0 出发,令 ve[0] = 0, 按照拓扑排序求其余各个顶点的最早发生时间 ve[i], (i <= i <= n-1)。如果得到的拓扑有序序列中顶点的个数小于网中的顶点数 n,则说明网中存在环,不能求关键路径,算法终止;否则执行步骤 3;
  3. 从汇点 vn 出发,令 vl[n-1] = ve[n-1],按照逆拓扑有序求其余各顶点的最迟发生时间 vl[i], (n-2 >= i >= 2);
  4. 根据各顶点的 ve 和 vl 值,求每条弧 s 的最早开始时间 e(s) 和最迟开始时间 l(s)。若某条弧满足条件 e(s) = l(s), 则为关键活动。

这篇关于DSA 经典数据结构与算法 学习心得和知识总结(三) |有向无环图及其应用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/716351

相关文章

Python中的随机森林算法与实战

《Python中的随机森林算法与实战》本文详细介绍了随机森林算法,包括其原理、实现步骤、分类和回归案例,并讨论了其优点和缺点,通过面向对象编程实现了一个简单的随机森林模型,并应用于鸢尾花分类和波士顿房... 目录1、随机森林算法概述2、随机森林的原理3、实现步骤4、分类案例:使用随机森林预测鸢尾花品种4.1

将Python应用部署到生产环境的小技巧分享

《将Python应用部署到生产环境的小技巧分享》文章主要讲述了在将Python应用程序部署到生产环境之前,需要进行的准备工作和最佳实践,包括心态调整、代码审查、测试覆盖率提升、配置文件优化、日志记录完... 目录部署前夜:从开发到生产的心理准备与检查清单环境搭建:打造稳固的应用运行平台自动化流水线:让部署像

Python中实现进度条的多种方法总结

《Python中实现进度条的多种方法总结》在Python编程中,进度条是一个非常有用的功能,它能让用户直观地了解任务的进度,提升用户体验,本文将介绍几种在Python中实现进度条的常用方法,并通过代码... 目录一、简单的打印方式二、使用tqdm库三、使用alive-progress库四、使用progres

Linux中Curl参数详解实践应用

《Linux中Curl参数详解实践应用》在现代网络开发和运维工作中,curl命令是一个不可或缺的工具,它是一个利用URL语法在命令行下工作的文件传输工具,支持多种协议,如HTTP、HTTPS、FTP等... 目录引言一、基础请求参数1. -X 或 --request2. -d 或 --data3. -H 或

在Ubuntu上部署SpringBoot应用的操作步骤

《在Ubuntu上部署SpringBoot应用的操作步骤》随着云计算和容器化技术的普及,Linux服务器已成为部署Web应用程序的主流平台之一,Java作为一种跨平台的编程语言,具有广泛的应用场景,本... 目录一、部署准备二、安装 Java 环境1. 安装 JDK2. 验证 Java 安装三、安装 mys

Python中构建终端应用界面利器Blessed模块的使用

《Python中构建终端应用界面利器Blessed模块的使用》Blessed库作为一个轻量级且功能强大的解决方案,开始在开发者中赢得口碑,今天,我们就一起来探索一下它是如何让终端UI开发变得轻松而高... 目录一、安装与配置:简单、快速、无障碍二、基本功能:从彩色文本到动态交互1. 显示基本内容2. 创建链

Node.js 中 http 模块的深度剖析与实战应用小结

《Node.js中http模块的深度剖析与实战应用小结》本文详细介绍了Node.js中的http模块,从创建HTTP服务器、处理请求与响应,到获取请求参数,每个环节都通过代码示例进行解析,旨在帮... 目录Node.js 中 http 模块的深度剖析与实战应用一、引言二、创建 HTTP 服务器:基石搭建(一

Android数据库Room的实际使用过程总结

《Android数据库Room的实际使用过程总结》这篇文章主要给大家介绍了关于Android数据库Room的实际使用过程,详细介绍了如何创建实体类、数据访问对象(DAO)和数据库抽象类,需要的朋友可以... 目录前言一、Room的基本使用1.项目配置2.创建实体类(Entity)3.创建数据访问对象(DAO

java中VO PO DTO POJO BO DO对象的应用场景及使用方式

《java中VOPODTOPOJOBODO对象的应用场景及使用方式》文章介绍了Java开发中常用的几种对象类型及其应用场景,包括VO、PO、DTO、POJO、BO和DO等,并通过示例说明了它... 目录Java中VO PO DTO POJO BO DO对象的应用VO (View Object) - 视图对象

Java向kettle8.0传递参数的方式总结

《Java向kettle8.0传递参数的方式总结》介绍了如何在Kettle中传递参数到转换和作业中,包括设置全局properties、使用TransMeta和JobMeta的parameterValu... 目录1.传递参数到转换中2.传递参数到作业中总结1.传递参数到转换中1.1. 通过设置Trans的