AOE网络的基本概念

上一节介绍了活动网络AOV网络的相关内容,这一节将进一步介绍另一种活动网络AOE网络。如果对于有向无环图(DAG),用有向边表示一个工程的各项活动(activity),边上的权值表示活动的持续时间duration),用顶点表示事件(event),那么这种DAG被称为边表示活动的网络Activity On Edges),简称AOE网络

图1

如图所示为一个AOE网络,可以看到有11项活动,有9个事件。事件发生表示之前的活动都已经完成,例如发生表示已完成,可以开始。每条边的权重表示对应活动的持续时间。工程开始之后,可以并行执行,而发生后,也可以并行执行。对于AOE网络,其有两个特殊的顶点:开始点(, 入度为0的顶点)称之为源点(source)和结束点(,出度为0的点)称之为汇点(sink),分别表示整个工程的开始和结束,均只有一个。

AOE网络主要要解决的问题:

1.完成整个工程至少需要多长时间?

2.为缩短完成工程的事件,应加快哪些活动?

可以看到从源点到汇点有多条路径,而由于并行化的原因,完成整个工程所需的时间取决于最长路径的长度,这条最长路径称为关键路径(critical path)。例如图1的关键路径为或者  ,持续时间之和都是18。

关键路径求解方法

关键量定义

找出关键路径可以分解为找出关键活动(critical activity),即关键路径上的所有活动。先定义几个关键量:

1):事件最早可能的开始时间表示顶点到顶点的最长路径长度,例如;

2):事件最迟允许开始时间,即在保证汇点时刻完成的前提下,事件的允许最迟开始时间,其等于减去的最长路径长度,例如;

3):活动的最早可能开始时间在有向边上,则是从源点到顶点的最长路径长度,因此;

4):活动的最迟允许开始时间在有向边上,则是在不会引起时间延迟的前提下,该活动允许的最迟开始时间。 ,为完成所需的时间。

表示活动的最早可能开始时间和最迟允许开始时间的时间余量,也称为松弛时间(slack time)。若则表示活动没有时间余量,是关键活动

看下图1的例子,对于:

是关键路径上的关键活动。

考虑:

可以推迟3个时间单位,并不是关键活动。

递推关系

为了找出关键活动,就需要求得各个活动的,从而判别二者是否相等。而要求得需要先求得各个顶点的最早可能开始时间和最迟允许开始时间。下面分别介绍求的递推公式。

1.求的递推公式,从开始,向前递推

为指向顶点得所有有向边的集合。

2.求的递推公式。从(有定义可知)开始,反向递推

是所有从顶点出发的有向边集合。

显然这两条递推公式可以通过之前的定义直接得到。

这两个递推公式的计算必须分别在拓扑排序和逆拓扑排序的前提下进行,逆拓扑排序(reverse topological sort)是指首先输出出度为0的顶点,以相反的次序输出拓扑排序序列。

按照拓扑排序,在计算时,的所有前驱顶点都已经求出。

按照逆拓扑排序,在计算时,的所有后继顶点都已经求出,这里我们需要根据拓扑排序计算的来计算

3.求的递推公式,活动对应带权有向边,则有

关键路径算法实现

根据前面分析,我们可以给出计算关键路径的算法

1)构建邻接表;

2)从源点出发,令,按照拓扑排序计算每个顶点的,若存在有向环则不能继续求关键路径;

3)从汇点出发,令,按照逆拓扑排序求各顶点的的

4)根据各顶点的,求各边的

5)每条边如果满足,则是关键活动,求出所有关键活动并输出。

代码实现

基于上一节AOV网络的代码来实现,同样用邻接表来表示连接,但需要增加连接的信息,每个连接的数据结构变为:

typedef struct Vertex{string name; //顶点名字int index; //顶点索引编号int dur; // 活动的持续时间int no; // 活动序号Vertex(string inputName, int inputIndex, int inputDur, int inputNo):name(inputName),index(inputIndex),dur(inputDur),no(inputNo){}Vertex():name(""),index(0),dur(0),no(0){}
} VertexNode;

增加活动持续时间和活动序号,另外图信息文件与之前有所不同,文件得每一样存储内容格式如下:起始顶点->连接顶点1->连接1持续时间->连接顶点2->连接2持续时间.....

我们一个例子(graph_struct.txt):

0 1 6 2 4 3 5
1 4 1
2 4 1
3 5 2
4 6 9 7 7
5 7 4
6 8 2
7 8 4
8

对应的就是图1中的图结构,完整代码如下:

#include<iostream>
#include<vector>
#include<fstream>
#include<sstream>
#include<string>
#include<unordered_map>
#include<stack>
using namespace std;
typedef struct Vertex{string name; //顶点名字int index; //顶点索引编号int dur; // 活动的持续时间int no; // 活动序号Vertex(string inputName, int inputIndex, int inputDur, int inputNo):name(inputName),index(inputIndex),dur(inputDur),no(inputNo){}Vertex():name(""),index(0),dur(0),no(0){}
} VertexNode;
// graph表示邻接表,id表示每个节点入度统计,topSortId存储拓扑排序的索引。
bool criticalPath(const vector<vector<VertexNode> >& graph, const vector<vector<VertexNode> >& revGraph, vector<int>& id, vector<int>& invId,vector<int>& Ee, vector<int>& El, vector<int>& e, vector<int>& L) {vector<int> topSortId;topSortId.reserve(graph.size());//step1: 拓扑排序,获取各事件最早可能开始时间Eestack<int> S; // 栈,存放入度为0的顶点for(int i = 0; i < graph.size(); i++) {if (id[i] == 0) {S.push(i);}}for (int i = 0; i < graph.size(); i++) {if(S.empty()) {return false;}int j = S.top();S.pop(); //弹出栈顶存储的顶点jtopSortId.push_back(j); //存入拓排序序列for(int k = 0; k < graph[j].size(); k++) {int currentIndex = graph[j][k].index; // 顶点j连接的顶点int currentDur = graph[j][k].dur; // 顶点j连接的顶点之间边的活动持续时间if(--id[currentIndex] == 0) {S.push(currentIndex);}if (Ee[j] + currentDur > Ee[currentIndex]) {Ee[currentIndex] = Ee[j] + currentDur;}}}// step2: 逆拓扑排序,获取各事件最迟允许开始时间Elstack<int> S1; // 栈,存放出度为0的顶点for(int i = 0; i < El.size(); i++) {El[i] = Ee[Ee.size() - 1];if(invId[i] == 0) S1.push(i); //初始出度为0的顶点入栈}for (int i = 0; i < revGraph.size(); i++) {int j = S1.top();S1.pop(); //弹出栈顶存储的顶点jfor(int k = 0; k < revGraph[j].size(); k++) {int currentIndex = revGraph[j][k].index; // 顶点j连接的顶点int currentDur = revGraph[j][k].dur; // 顶点j连接的顶点之间边的活动持续时间if(--invId[currentIndex] == 0) {S1.push(currentIndex);}if (El[j] - currentDur < El[currentIndex]) {El[currentIndex] = El[j] - currentDur;}}}// step3: 判定各条边是否是关键活动for(int i = 0; i < graph.size(); i++) {for(int k = 0; k < graph[i].size(); k++) {int currentIndex = graph[i][k].index; // 顶点j连接的顶点int currentDur = graph[i][k].dur; // 顶点j连接的顶点之间边的活动持续时间int no = graph[i][k].no; // 该条边对应活动的序号e[no] = Ee[i];L[no] = El[currentIndex] - currentDur;if(e[no] == L[no]) {cout << "a" << no + 1 <<" : " << i << "->" << currentIndex <<endl;}}}return true; // 如果前面所有顶点全部循环没问题,那么说明可以拓扑排序故返回true.
}
int main() {unordered_map<string ,int> graphMap; // 图节点名和编号的Mapvector<vector<VertexNode> > adjGraph; // 图的连接表表示法,邻接表vector<vector<VertexNode> > reverseAdjGraph; // 图的连接表表示法,逆邻接表ifstream graphRdFile("graph_struct.txt");if(!graphRdFile.good()) {cout << "open graph file failed!" << endl;return -1;}string line;int index = 0;string vertexName;// 首先对Vertex Name进行编码while (getline(graphRdFile, line)) {istringstream ss(line);string tmp1,tmp2; //顶点名字和顶点权重。if (ss >> tmp1) {if (graphMap.find(tmp1) == graphMap.end()) {graphMap.insert(make_pair(tmp1, index++));}}while(ss >> tmp1 >> tmp2) {if (graphMap.find(tmp1) == graphMap.end()) {graphMap.insert(make_pair(tmp1, index++));}}}// 编码与Vertex的反映射vector<string> indexName = vector<string>(graphMap.size(),"");for(auto itr=graphMap.begin();itr!=graphMap.end();itr++) {indexName[itr->second] = itr->first;}// 重新读graphRdFile.clear();graphRdFile.seekg(0,std::ios::beg);adjGraph.resize(graphMap.size());reverseAdjGraph.resize(graphMap.size());int currentIndex = 0; // 当前图节点的编号vector<int> id(adjGraph.size(), 0); // 每个节点入度统计vector<int> invId(adjGraph.size(), 0); // 每个节点出度统计vector<int> Ee(adjGraph.size(), 0); //各事件最早可能开始时间vector<int> El(adjGraph.size(), 0); //各事件最迟允许开始时间int activNum = 0; // 活动序号while (getline(graphRdFile, line)) { //按行读,每一行是一个图节点的连接情况istringstream ss(line);string tmp; // 顶点名字int inputDur; //顶点权重bool firstFlag = true;while(ss >> tmp) {if (firstFlag) {if (graphMap.find(tmp) != graphMap.end()) {currentIndex = graphMap[tmp];} else {break;}firstFlag = false;continue;}if (graphMap.find(tmp) != graphMap.end()) {if(ss >> inputDur){adjGraph[currentIndex].emplace_back(VertexNode(tmp,graphMap[tmp],inputDur, activNum++)); //邻接表构造id[graphMap[tmp]]++; // 终点入度+1invId[currentIndex]++; //起点出度+1reverseAdjGraph[graphMap[tmp]].emplace_back(VertexNode(indexName[currentIndex],currentIndex,inputDur,0));}}}}vector<int> e(activNum, 0);  //各活动最早可能开始时间vector<int> L(activNum, 0);  //各活动最迟允许开始时间// AOE关键路径测试:if(criticalPath(adjGraph, reverseAdjGraph, id, invId, Ee, El, e, L)) {cout << "Success!" << endl;} else {cout << "Network has a cycle!" << endl;}return 0;
}

对应图1的图结构执行结果如下:

a1 : 0->1
a4 : 1->4
a7 : 4->6
a8 : 4->7
a10 : 6->8
a11 : 7->8
Success!

可以看到成功输出了所有的关键活动。

图算法入门4:活动网络-AOE网络和关键路径(critical path)相关推荐

  1. AOE网络关键路径求解的通俗理解

    AOE网络是一种以边主导的活动网络,简单来说是一种有向加权图,通常有一个起点(源点)和一个终点(汇点),中间顶点表示某个事件,或叫里程碑,而有向边叫做活动,AOE网络通常可以用来表达某些带有前驱后继关 ...

  2. AOV网络和AOE网络

    AOV和AOE网络是什么 活动网络可以用来描述生产计划.施工过程.生产流程.程序流程等工程中各子工程的安排问题.活动网络可分为两种:AOV网络和AOE网络 AOV网络(Activity On Vert ...

  3. 【数据结构笔记30】拓扑排序、AOV网络、AOE网络、关键工序

    本次笔记内容: 8.2.1 拓扑排序 8.2.2 关键路径 文章目录 拓扑排序 引例:计算机专业排课(AOV网络) 拓扑排序的概念与AOV网络 拓扑排序的算法 更加聪明的算法 关键路径问题 AOE(A ...

  4. 【数据结构与算法基础】AOE网络与关键路径

    前言 数据结构,一门数据处理的艺术,精巧的结构在一个又一个算法下发挥着他们无与伦比的高效和精密之美,在为信息技术打下坚实地基的同时,也令无数开发者和探索者为之着迷. 也因如此,它作为博主大二上学期最重 ...

  5. AOV网络与AOE网络

    一.AOV网络与拓扑排序   AOV网(Activity On Vertex NetWork)用顶点表示活动,边表示活动(顶点)发生的先后关系.   若网中所有活动均可以排出先后顺序(任两个活动之间均 ...

  6. 网络编程懒人入门(二):快速理解网络通信协议(下篇)

    1.前言 本文上篇<网络编程懒人入门(一):快速理解网络通信协议(上篇)>分析了互联网的总体构思,从下至上,每一层协议的设计思想.基于知识连贯性的考虑,建议您先看完上篇后再来阅读本文. 本 ...

  7. 网络编程懒人入门(一):快速理解网络通信协议(上篇)

    1.写在前面 论坛和群里常会有技术同行打算自已开发IM或者消息推送系统,很多时候连基本的网络编程理论(如网络协议等)都不了解,就贸然定方案.写代码,显得非常盲目且充满技术风险. 即时通讯网论坛里精心整 ...

  8. 石头机器人红灯快闪_机器人集体“快闪”活动爆红网络 “我是AI”与您相约智能新时代...

    原标题:机器人集体"快闪"活动爆红网络 "我是AI"与您相约智能新时代 3月10日下午,天津科学技术馆内,悠扬美妙的歌声<我和我的祖国>突然响起,随 ...

  9. PyTorch系列入门到精通——生成对抗网络一瞥

    PyTorch系列入门到精通--生成对抗网络一瞥

最新文章

  1. spring aop实现过程之一代理对象的生成
  2. 屏蔽朋友圈的第一天的感悟
  3. 一个详尽的面向 SAP UI5 初学者的教程 - 如何在 SAP UI5 中绘制图表 Chart
  4. 用户体验岗如何说服其他部门_为什么我们应该说服用户更新他们的浏览器-这是双赢的。...
  5. app嵌入jsp页面的项目工作量_好程序员Java学习路线分享jsp为什么用的不多了
  6. Atitit 架构之道 attilax著 1. 架构的目的是什么??提高架构可读性。。提高扩展性。。对兼容性也有一定提升。。 3 1.1. 伸缩性架构设计 3 1.2. 提升性能架构 3 1.3.
  7. Atitit.播放系统的选片服务器,包厢记时系统 的说明,教程,维护,故障排查手册p825
  8. DSP28335学习记录(三)——ePWM
  9. 政务信息系统整合共享实施方案
  10. 【论文精读】A view-free image stitching network based on global homography-基于全局单应的无视图图像拼接网络
  11. 边沿触发是什么意思_边沿触发器的动作特点及主要特点
  12. MongoDB数据导入 JSON
  13. win10连接文件服务器记住密码如何删除,win10系统访问共享文件夹需要密码的取消方法...
  14. handsontable 给单元格设置下拉 菜单
  15. IT行业就业前景如何
  16. php gd imageaffine
  17. 相机响应曲线、ISO详解
  18. python归一化 增大差异_Python实现描述性统计
  19. python 老照片修复软件_这款开源的 Python 老照片修复工具火了
  20. SpringBoot访问静态资源(图片)

热门文章

  1. Kotlin/DSL(Anko),原汁原味Kotlin开发Android---Activity Fragment与AnkoUI分离,强大的复用,更加便捷的开发
  2. 【解决Hibernate异常 identifier of an instance of xxx(实体类) was altered from xxx to xxxPK】
  3. 关键链方法的多项目监控技术
  4. 谷歌学术、中国知网生成参考文献
  5. opencv mat裁剪
  6. Java 多态的薪酬计算的练习
  7. mPaaS客户端基线升级踩坑先升到10.1.32再升到10.1.60(iOS)
  8. Loadrunner11安装及globa-10000配置
  9. Linux TCP在3.18内核引入的一个慢启动相关的问题或者说Bug
  10. simpread-PCB 基本布线规范与设计原则