🔥 本文由 程序喵正在路上 原创,CSDN首发!
💖 系列专栏:数据结构与算法
🌠 首发时间:2022年12月4日
🦋 欢迎关注🖱点赞👍收藏🌟留言🐾
🌟 一以贯之的努力 不得懈怠的人生
阅读指南
- 有向无环图描述表达式
- 有向无环图(DAG)
- DAG 描述表达式
- 解题方法
- 拓扑排序
- AOV网
- 拓扑排序
- 逆拓扑排序
- 逆拓扑排序的实现(DFS算法)
- 关键路径
- AOE网
- 求关键路径的步骤
- 关键活动、关键路径的特性
有向无环图描述表达式
有向无环图(DAG)
如果一个有向图中不存在环,则称之为有向无环图,简称 D A G DAG DAG 图( D i r e c t e d A c y c l i c G r a p h Directed \ Acyclic \ Graph Directed Acyclic Graph),比如下面这个图
DAG 描述表达式
有下面这样一个表达式:
( ( a + b ) ∗ ( b ∗ ( c + d ) ) + ( c + d ) ∗ e ) ∗ ( ( c + d ) ∗ e ) ((a + b) * (b * (c + d)) + (c + d) * e) * ((c + d) * e) ((a+b)∗(b∗(c+d))+(c+d)∗e)∗((c+d)∗e)
我们可以将其表示成树的形式:
但是我们会发现,树中有些地方是重复的,比如 ( c + d ) ∗ e (c + d) * e (c+d)∗e 这部分,所以我们删除一部分,变成
接着我们发现还有 c + d c + d c+d 这部分也重复了,也得删去
最后还有 b b b 也重复
规律:顶点中不可能出现重复的操作数
解题方法
-
把各个操作数不重复地排成一排
-
标出每个运算符的生效顺序(先后顺序有点出入无所谓)
-
按顺序加入运算符,注意 “分层”
-
从底向上逐层检查同层的运算符是否可以合体
拓扑排序
AOV网
A O V AOV AOV 网( A c t i v i t y O n V e r t e x N e t W o r k Activity \ On \ Vertex \ NetWork Activity On Vertex NetWork,用顶点表示活动的网):用 D A G DAG DAG 图(有向无环图)表示一个工程。顶点表示活动,有向边 < V i , V j > <V_i, V_j> <Vi,Vj> 表示活动 V i V_i Vi 必须先于活动 V j V_j Vj 进行
拓扑排序
在图论中,由一个有向无环图的顶点组成的序列,当且仅当满足下列条件时,称为该图的一个拓扑排序:
- 每个顶点出现且只出现一次
- 若顶点 A A A 在序列中排在顶点 B B B 的前面,则在图中不存在从顶点 B B B 到顶点 A A A 的路径
每个 A O V AOV AOV 网都有一个或多个拓扑排序序列
拓扑排序的实现:
- 从 A O V AOV AOV 网中选择一个没有前驱(入度为 0 0 0)的顶点并输出
- 从网中删除该顶点和所有以它为起点的有向边
- 重复 1 1 1 和 2 2 2 直到当前的 A O V AOV AOV 网为空或者当前网中不存在无前驱的顶点为止
#define MaxVertexNum 100 //图中顶点数目的最大值
typedef struct ArcNode { //边表结点
int adjvex; //该弧所指向的顶点的位置
struct ArcNode *nextarc; //指向下一条弧的指针
//InfoType info; //网的边权值
}ArcNode;
typedef struct VNode { //顶点表结点
VertexType data; //顶点信息
ArcNode *firstarc; //指向第一条依附该顶点的弧的指针
}VNode, AdjList[MaxVertexNum];
typedef struct {
AdjList vertices; //邻接表
int vexnum, arcnum; //图的顶点数和弧数
}Graph; //Graph是以邻接表存储的图类型
bool TopologicalSort(Graph G) {
InitStack(S); //初始化栈,存储入度为0的顶点
for (int i = 0; i < G.vexnum; ++i) {
if (indegree[i] == 0) Push(S, i); //将所有入度为0的顶点进栈
}
int count = 0; //计数,记录当前已经输出的顶点数
while (!IsEmpty(S)) { //栈不空,则存在入度为0的顶点
Pop(S, i); //栈顶元素出栈
print[count++] = i; //输出顶点i
for (p = G.vertices[i].firstarc; p; p = p->nextarc) {
//将所有i指向的顶点的入度减1,并且将入度减为0的顶点压入栈S
v = p->adjvex;
if (!(--indegree[v])) Push(S, v); //入度为0,则入栈
}//for
}//while
if (count < G.vexnum) return false; //排序失败,有向图中有回路
else return true; //拓扑排序成功
其中 indegree
是记录各个顶点入度的数组,初始化为各顶点的入度;print
是记录拓扑序列的数组,初始化为
−
1
-1
−1
在这个拓扑排序中,每个顶点和每条边都需要处理一次,整体的时间复杂度为 O ( ∣ V ∣ + ∣ E ∣ ) O(|V| + |E|) O(∣V∣+∣E∣),如果是采用邻接矩阵存储的图,则需要 O ( ∣ V ∣ 2 ) O(|V|^2) O(∣V∣2) 的时间复杂度
逆拓扑排序
逆拓扑排序的实现:
- 从 A O V AOV AOV 网中选择一个没有后继(出度为 0 0 0)的顶点并输出
- 从网中删除该顶点和所有以它为终点的有向边
- 重复 1 1 1 和 2 2 2 直到当前的 A O V AOV AOV 网为空
逆拓扑排序的实现(DFS算法)
void DFSTraverse(Graph G) { //对图G进行深度优先遍历
for (v = 0; v < G.vexnum; ++v) //初始化已访问标记数据
visited[v] = false;
for (v = 0; v < G.vexnum; ++v)
if (!visited[v]) DFS(G, v);
}
void DFS(Graph G, int v) { //从顶点v开始,深度优先遍历图G
visited[v] = true; //设置已访问标记
for (w = FirstNeighbor(G, v); w >= 0; w = NextNeighbor(G, v, w))
if (!visited[w]) DFS(G, w);
print(v); //输出顶点
}
如果图中存在回路,那么上面的程序就不再使用,我们需要给每个顶点添加一个是否已经入栈的标记来判断是否有回路即可
关键路径
AOE网
在带权有向图中,以顶点表示事件,以有向边表示活动,以边上的权值表示完成该活动的开销(如完成该活动所需的时间),称之为哟个边表示活动的网络,简称 A O E AOE AOE 网( A c t i v i t y O n E d g e N e t W o r k Activity \ On \ Edge \ NetWork Activity On Edge NetWork)
A O E AOE AOE 网具有以下两个性质:
- 只有在某顶点所代表的事件发生后,从该顶点出发的各有向边所代表的活动才能开始
- 只有在进入某顶点的各有向边所代表的活动都已结束时,该顶点所代表的事件才能发生。另外,有些活动是可以并行进行的
在 A O E AOE AOE 网中仅有一个入度为 0 0 0 的顶点,称为开始顶点(源点),它表示整个工程的开始;也仅有一个出度为 0 0 0 的顶点,称为结束顶点(汇点),它表示整个工程的结束
从源点到汇点的有向路径可能有多条,所有路径中,具有最大路径长度的路径称为关键路径,而把关键路径上的活动称为关键活动
完成整个工程的最短时间就是关键路径的长度,若关键活动不能按时完成,则整个工程的完成时间就会延长
事件 v k v_k vk 的最早开始时间 v e ( k ) ve(k) ve(k) —— 决定了所有从 v k v_k vk 开始的活动能够开工的最早时间
活动 a i a_i ai 的最早开始时间 e ( i ) e(i) e(i) —— 指该活动弧的起点所表示的事件的最早发生时间
事件 v k v_k vk 的最迟开始时间 v l ( k ) vl(k) vl(k) —— 它是指在不推迟整个工程完成的前提下,该事件最迟应该发生的时间
活动 a i a_i ai 的最迟开始时间 l ( i ) l(i) l(i) —— 它是指该活动弧的终点所表示事件的最迟发生时间与该活动所需时间之差
活动 a i a_i ai 的时间余量 d ( i ) = l ( i ) − e ( i ) d(i) = l(i) - e(i) d(i)=l(i)−e(i),表示在不增加完成整个工程所需总时间的情况下,活动 a i a_i ai 可以拖延的时间;若一个活动的时间余量为零,则说明该活动必须要如期完成, d ( i ) = 0 d(i) = 0 d(i)=0 即 l ( i ) = e ( i ) l(i) = e(i) l(i)=e(i) 的活动是关键活动,由关键活动组成的路径就是关键路径
求关键路径的步骤
- 求所有事件的最早发生时间 v e ( ) ve() ve()
- 求所有事件的最迟发生时间 v l ( ) vl() vl()
- 求所有活动的最早发生时间 e ( ) e() e()
- 求所有活动的最迟发生时间 l ( ) l() l()
- 求所有活动的时间余量 d ( ) d() d()
① 求所有事件的最早发生时间 v e ( ) ve() ve()
按拓扑排序序列,依次求各个顶点的 v e ( k ) ve(k) ve(k):
- v e ve ve( 源点 ) = 0 = 0 =0
- v e ( k ) = M a x { v e ( j ) + W e i g h t ( v j , v k ) } ve(k) = Max\{ve(j) + Weight(v_j, v_k)\} ve(k)=Max{ve(j)+Weight(vj,vk)}, v j v_j vj 为 v k v_k vk 的任意前驱
上图的拓扑序列为: V 1 、 V 3 、 V 2 、 V 5 、 V 4 、 V 6 V_1、V_3、V_2、V_5、V_4、V_6 V1、V3、V2、V5、V4、V6
那么
v e ( 1 ) = 0 ve(1) = 0 ve(1)=0
v e ( 3 ) = 2 ve(3) = 2 ve(3)=2
v e ( 2 ) = 3 ve(2) = 3 ve(2)=3
v e ( 5 ) = 6 ve(5) = 6 ve(5)=6
v e ( 4 ) = m a x { v e ( 2 ) + 2 , v e ( 3 ) + 4 } = 6 ve(4) = max\{ve(2) + 2, ve(3) + 4\} = 6 ve(4)=max{ve(2)+2,ve(3)+4}=6
v e ( 6 ) = m a x { v e ( 5 ) + 1 , v e ( 4 ) + 2 , v e ( 3 ) + 3 } = 8 ve(6) = max\{ve(5) + 1, ve(4) + 2, ve(3) + 3\} = 8 ve(6)=max{ve(5)+1,ve(4)+2,ve(3)+3}=8
② 求所有事件的最迟发生时间 v l ( ) vl() vl()
按逆拓扑排序序列,依次求各个顶点的 v l ( k ) vl(k) vl(k):
- v l vl vl(汇点) = = = v e ve ve(汇点)
- v l ( k ) = M i n { v l ( j ) + W e i g h t ( v k , v j ) } vl(k) = Min\{vl(j) + Weight(v_k, v_j)\} vl(k)=Min{vl(j)+Weight(vk,vj)}, v j v_j vj 为 v k v_k vk 的任意后继
上图的逆拓扑序列为: V 6 、 V 5 、 V 4 、 V 2 、 V 3 、 V 1 V_6、V_5、V_4、V_2、V_3、V_1 V6、V5、V4、V2、V3、V1
那么
v l ( 6 ) = 8 vl(6) = 8 vl(6)=8
v l ( 5 ) = 7 vl(5) = 7 vl(5)=7
v l ( 4 ) = 6 vl(4) = 6 vl(4)=6
v l ( 2 ) = m i n { v l ( 5 ) − 3 , v l ( 4 ) − 2 } = 4 vl(2) = min\{vl(5) - 3, vl(4) - 2\} = 4 vl(2)=min{vl(5)−3,vl(4)−2}=4
v l ( 3 ) = m i n { v l ( 4 ) − 4 , v l ( 6 ) − 3 } = 2 vl(3) = min\{vl(4) - 4, vl(6) - 3\} = 2 vl(3)=min{vl(4)−4,vl(6)−3}=2
v l ( 1 ) = m i n { v l ( 2 ) − 3 , v l ( 3 ) − 2 } = 0 vl(1) = min\{vl(2) - 3, vl(3) - 2\} = 0 vl(1)=min{vl(2)−3,vl(3)−2}=0
③ 求所有活动的最早发生时间 e ( ) e() e()
若边 < v k , v j > <v_k, v_j> <vk,vj> 表示活动 a i a_i ai,则有 e ( i ) = v e ( k ) e(i) = ve(k) e(i)=ve(k),说白了就等于活动 a i a_i ai 所在弧的弧尾的事件的最早发生时间 v e ( k ) ve(k) ve(k),所以
e ( 1 ) = v e ( 1 ) = 0 , e ( 2 ) = v e ( 1 ) = 0 , e ( 3 ) = v e ( 2 ) = 3 , e ( 4 ) = 4 , e ( 5 ) = v e ( 3 ) = 2 , e ( 6 ) = 2 , e ( 7 ) = 6 , e ( 8 ) = 6 e(1) = ve(1) = 0, \ e(2) = ve(1) = 0, \ e(3) = ve(2) = 3, \ e(4) = 4, \ e(5) = ve(3) = 2, \ e(6) = 2, \ e(7) = 6, \ e(8) = 6 e(1)=ve(1)=0, e(2)=ve(1)=0, e(3)=ve(2)=3, e(4)=4, e(5)=ve(3)=2, e(6)=2, e(7)=6, e(8)=6
④ 求所有活动的最迟发生时间 l ( ) l() l()
若边 < v k , v j > <v_k, v_j> <vk,vj> 表示活动 a i a_i ai,则有 l ( i ) = v l ( j ) − W e i g h t ( v k , v j ) l(i) = vl(j) - Weight(v_k, v_j) l(i)=vl(j)−Weight(vk,vj),所以
l ( 1 ) = v l ( 2 ) − 3 = 1 , l ( 2 ) = v l ( 3 ) − 2 = 0 , l ( 3 ) = 4 , l ( 4 ) = 4 , l ( 5 ) = 2 , l ( 6 ) = 5 , l ( 7 ) = 6 , l ( 8 ) = 7 l(1) = vl(2) - 3 = 1, \ l(2) = vl(3) - 2 = 0, \ l(3) = 4, \ l(4) = 4, \ l(5) = 2, \ l(6) = 5, \ l(7) = 6, \ l(8) = 7 l(1)=vl(2)−3=1, l(2)=vl(3)−2=0, l(3)=4, l(4)=4, l(5)=2, l(6)=5, l(7)=6, l(8)=7
⑤ 求所有活动的时间余量 d ( ) d() d()
d ( i ) = l ( i ) − e ( i ) d(i) = l(i) - e(i) d(i)=l(i)−e(i)
方面 | a 1 a_1 a1 | a 2 a_2 a2 | a 3 a_3 a3 | a 4 a_4 a4 | a 5 a_5 a5 | a 6 a_6 a6 | a 7 a_7 a7 | a 8 a_8 a8 |
---|---|---|---|---|---|---|---|---|
e ( k ) e(k) e(k) | 0 | 0 | 3 | 3 | 2 | 2 | 6 | 6 |
l ( k ) l(k) l(k) | 1 | 0 | 4 | 4 | 2 | 5 | 6 | 7 |
d ( k ) d(k) d(k) | 1 | 0 | 1 | 1 | 0 | 3 | 0 | 1 |
关键活动: a 2 、 a 5 、 a 7 a_2、a_5、a_7 a2、a5、a7
关键路径: V 1 − > V 3 − > V 4 − > V 6 V_1 -> V_3 -> V_4 -> V_6 V1−>V3−>V4−>V6
关键活动、关键路径的特性
- 若关键活动耗时增加,则整个工程的工期将增长
- 缩短关键活动的时间,可以缩短整个工程的工期
- 当缩短到一定程度时,关键活动可能会变成非关键活动
- 可能有多条关键路径,只提高一条关键路径上的关键活动并不能缩短整个工程的工期,只有加快那些包括在所有关键路径上的关键活动才能达到缩短工期的目的