数据结构与算法:第4章 图以及与图有关的算法_第1页
数据结构与算法:第4章 图以及与图有关的算法_第2页
数据结构与算法:第4章 图以及与图有关的算法_第3页
数据结构与算法:第4章 图以及与图有关的算法_第4页
数据结构与算法:第4章 图以及与图有关的算法_第5页
已阅读5页,还剩72页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

4.1基本术语4.2图的表示

4.3图的搜索算法4.4图与树的联系

4.5无向图的双连通性*4.6有向图的搜索

4.7强连通图

4.8拓扑分类*4.9关键路径

4.10单源最短路径*4.11每一对顶点间的最短路径第4章图以及与图有关的算法4.1基本定义/术语[定义]一个图G=(V,E)是一个由非空的有限集V和一个边集E所组成的。若E中的每条边都是顶点的有序对(v,w),就说该图是有向图(directedgraph,digraph)。若E中的每条边是两个不同顶点无序对,就说该图是无向图,其边仍表示成(v,w)。[ADT]GraphG=(V,R)

数据对象v:v是具有相同特性的数据元素的集合,称为顶点集。数据关系R:

R={VR}VR={<v,w>|v,w∈V,且P(v,w),<v,w>表示从v到w的弧,谓词P(v,w)定义了弧<v,w>的意义或信息}操作:NodeNewNode(G,v)VoidDelNode(G,v)VoidSetSucc(G,v1,v2)VoidDelSucc(G,v1,v2)ListofnodeSucc(G,v)LisyofnodePred(G,v)IntIsEdge(G,v1,v2)NodeFirstAdjVex(G,v)NodeNextAdjVex(G,v,w)术语:顶点弧/边邻接/相邻依附路径(路)简单路径环路带标号的图(网)

连通连通图

强连通图连通分量完全图稀疏图稠密图子图度入度出度生成树V3V1V4V5V2G2V1V3V4V2G10110000000011000G1.arcs0101010101010111010001100G2.arcs126543545983516∞3∞∞∞1∞∞5∞3∞∞∞∞4∞∞98∞∞∞∞6∞∞5∞∞∞∞∞∞∞534.2图的表示1、图的顺序存储——邻接矩阵设图G=(V,E),V={0,1,…,n-1}则表示G的邻接矩阵A是其元素按下式定义的n*n矩阵:A[i][j]=1若(i,j)∈E0若(i,j)∈E网的邻接矩阵可定义为:A[i][j]=wij

若(i,j)∈E∞

若(i,j)∈ETD(vi)=∑A[i][j]=∑A[j][i](n顶点个数)j=0n-1j=0n-1TD(vi)=OD(vi)+ID(vi)=∑A[i][j]+∑A[j][i](n顶点个数)j=0n-1j=0n-1#defineINFINITYINT_MAX#defineMAX_VERTEX_NUM20Typedefenum{DG,DN,AG,AN}GraphKind;TypedefstructArcCell{VRTypeadj;InfoType*info;}ArcCell,AdjMatrix[MAX_VERTEX_NUM][MAX_VERTEX_NUM];Typedefstruct{VertexTypevex[MAX_VERTEX_NUM];AdjMatrixarcs;intvexnum,arcnum;

GraphKindkind;}Mgraph;例:图类型变量:MgraphG;顶点个数:G.vexnum弧/边的个数:G.arcnum图的类型:G.kind=(DG,DN,AG,AN)顶点i信息:G.vex[i]顶点i和顶点j邻接关系:G.arcs[i][j].adj弧/边附加信息:G.arcs[i][j].info->v1

v2

v3

v4

2130^^^有向图G2邻接表0123v1

v2

v3

v4

3002^^^^G2的逆邻接表0123无向图G1邻接表v5

v1

v2

v3

v4

314243202101^^^^^01234V3V1V4V5V2G1V1V3V4V2G2+2、图的链式存储——邻接表(AdjacencyList)2、图的链式存储——邻接表(AdjacencyList)Adjvexnextarcinfo表结点datafirstarc头结点#defineMAX_VERTEX_NUM20TypedefstructArcNode{intadjvex;structArcNode*nextarc;InfoType*info;}ArcNode;TypedefstructVnode{VertexTypedata;ArcNode*firstarc;}Vnode,AdjList[MAX_VERTEX_NUM];Typedefstruct{AdjListvertices;Intvexnum;Intkind;}ALGraph;例:图类型变量:ALgraphG;顶点个数:G.vexnum图的类型:G.kind=(DG,DN,AG,AN)顶点i信息:G.vertices[i].dada顶点i的第一个邻接点:G.vertices[i].firstarc->adjvexG.vertices[G.vertices[i].firstarc->adjvex].dataG.vertices[i].firstarc->info顶点i的第二个邻接点:G.vertices[i].firstarc->nextarc->adjvex4.3图的搜索算法深度优先搜索DFS(Depth-Firstsearch)广度优先搜索BFS(Breadth-Firstsearch)图的遍历确定遍历起点为保证非连通图的每一顶点都能被访问到,应轮换起点为避免顶点的重复访问,做访问标记遍历图注意问题:1、深度优先搜索DFS(Depth-Firstsearch)

首先访问起点,然后依次访问与该起点相关联的每一个顶点,并以该关联顶点为新的起点,继续深度优先遍历。若图中还有未被访问的顶点,则换一个起点,继续深度优先遍历;直到所有的顶点都被访问。V1V2V3V4V5V6V7V8无向图G3V1V2V3V4V5V6V7V8有向图G4V1,V3,V6,V7,V2,V4,V8,V5V1,V2,V4,V8,V5,V3,V6,V7深度优先遍历:VoidDFSTravers(GRAPHG,v){For(v=0;v<G.vexnum;++v)visited[v]=FALSE;For(v=0;v<G.vexnum;++v)if(!visited[v])DFS(G,v);}VoidDFS(GRAPHG,intv){visited[v]=TRUE;visitfunc(v);for(w=FirstAdjVex(G,v);w;w=NextAdjVex(G,v,w);if(!visited[w])DFS(G,w);}v13v2v3v4v54422123010101234^^无向图G2邻接表V3V1V4V5V2G2图G2的深度优先遍历结果:V1,V4,V3,V5,V2DFSTraverse(G){

T=¢;/*树边集开始为空*/count=1;/*先深编号计数器*/for(allv∈V)markv“new”while(thereexistsavertexv∈Vmarked“new”)search(v)}

voidsearch(v){dfn[v]=count;/*对v编号*/count=count+1;markv“old”/*访问结点v*/for(eachvertexw∈L[v])if(wismarked“new”{add(v,w)toT;/*(v,w)是树边*/search(w);/*递归搜索*/}}输入:L[v]表示无向图G的关于v的邻接表输出:每个结点有先深编号的无向图和树边集T2、广度优先搜索BFS(Breadth-Firstsearch)V1V2V3V4V5V6V7V8无向图G3V1V2V3V4V5V6V7V8有向图G4V1,V2,V3,V4,V5,V6,V7,V8V1,V2,V3,V4,V5,V6,V7,V8

首先访问起点,依次访问与该起点相关联的每一个邻接点,然后分别从这些邻接点出发访问它们的邻接点,并使“先被访问的顶点的邻接点”先于“后被访问的顶点的邻接点”被访问,若图中还有未被访问的顶点,则换一个起点,继续广度优先遍历;直到所有的顶点都被访问。VoidBFSTravers(GRAPHG,v){For(v=0;v<G.vexnum;++v)visited[v]=FALSE;InitQueue(Q);For(v=0;v<G.vexnum;++v)if(!visited[v]){EnQueue(Q,v);while(!QueueEmpty(Q)){DeQueue(Q,u);visited[u]=TRUE;visit(u);for(w=FirstAdjVex(G,u);w;w=NextAdjVex(G,u,w);if(!visited[w])EnQueue(Q,w);}}}v13v2v3v4v54422123010101234^^无向图G2邻接表V3V1V4V5V2G2图G2的广度优先遍历结果:V1,V4,V2,V3,V5Voidbsearch(v){MAKENULL(Q);bfn[v]=count;/*先广编号*/count=count+1;markv“old”ENQUEUE(v,Q);/*v入队*/while(!EMPTY(Q)){v=FRONT(Q);DEQUEUE(Q);for(eachw∈L[v]){bfn[w]=count;/*先广编号*/count=count+1;markw“old”;ENQUEUE(w,Q);INSERT((v,w),T);}}/*树边入队*/}输入:L[v]表示无向图G的关于v的邻接表输出:每个结点有先广编号的无向图和树边集T4.4图与树的联系4.4.1先深生成森林和先广生成森林abdecfg(a)abdecfg(b)abdecfg(c)树边(编号小->大)与非树边:回退边/横边(编号大->小)先深(广)搜索过程中,由树边和树边所连接的顶点组成的子图,称为图的先深(广)生成森林。无向连通图通过先深(广)搜索只能得到一个先深(广)生成树非连通图则可得到多个生成树连通子图(连通分量)4.4.3最小生成树4.4.2无向图与开放树[定义]

连通而无环路的无向图称作开放树(FreeTree)开放树的性质:(1)具有n≥1个顶点的开放树包含n-1条边(2)如果在开放树中任意加上一条边,便得到一条回路(证明见教材P154~155)

设G=(V,E)是一个连通图,E中每一条边(u,v)的权为C(u,v),也叫做边长。图G的一株生成树(spanningtree)是连接V中所有结点的一株开放树。将生成树中所有边长之总和称为生成树的价(cost)。使这个价最小的生成树称为图G的最小生成树(minimum-costspanningtree)。MST性质

设G=(V,E)是一个连通图,在E上定义一个权函数,且{(V1,T1),(V2,T2),…,(Vk,Tk)}是G的任意生成森林。令T=∪Ti(k>1)ki=1

又设e=(v,w)是E-T中这样一条边,其权C[v,w]最小,而且v∈V1和w∈V1。则图G有一棵包含T∪{e}的生成树,其价不大于包含T的任何生成树的价。

假设N=(V,{E})是一个连通网,U是顶点V的一个非空子集。若(u,v)是一条具有最小权值(代价)的边,其中u∈U,v∈V-U,则必存在一棵包含边(u,v)的最小生成树。描述1:描述2:UV-Uuvu’v’MST性质证明假设网N的任何一棵最小生成树都不包含(u,v),设T是连通网上的一棵最小生成树,当将边(u,v)加入到T中时,由生成树的定义,T中必包含一条(u,v)的回路。另一方面,由于T是生成树,则在T上必存在另一条边(u’,v’),且u和u’、v和v’之间均有路径相同。删去边(u’,v’)便可消去上述回路,同时得到另一棵最小生成树T’。但因为(u,v)的代价不高于(u’,v’),则T’的代价亦不高于T,T’是包含(u,v)的一棵最小生成树。(a)V3V1V4V5V2V66515536462V3V1V4V5V2(c)V614V3V1V4V5V2(d)V6142V3V1V4V5V2(e)V61542V3V1V4V5V2(f)V615342求最小生成树V3V1V4V5V2(b)V61UV-U1、求最小生成树——Prim算法输入:加权无向图(无向网)G=(V,E),其中v=(1,2,…,n).输出:G的最小生成树要点:引入集合U和T。U准备结点集,T为树边集。初值U={1},

T=¢。选择有最小权的边(u,v),使u∈U,v∈(V-U),将v加入

U,(u,v)加入T。重复这一过程,直到U=V.voidprim(G,T){T=¢;U={1};while((U–V)!=¢){设(u,v)是使u∈U与v∈(V-U)且权最小的边;

T=T∪{(u,v)};U=U∪{v};}};引入辅助向量:

CLOSEST[]和LOWCOSTCLOSEST[i]为U中的一个顶点则边(i,CLOSEST[i])具有最小的权LOWCOST[i]集合如何实现?若顶点i∈U则LOWCOST[i]=INFINITYCLOSTST和LOWCOST的初值是多少?开始调整LOWCOST,对非集合U中的顶点jLOWCOST[j]=min(C[k][j],LOWCOST[j])CLOSEST[j]=k;i++结束LOWCOST[i]为邻接矩阵C的第一行值CLOSEST[i]的初值均为1从LOWCOST[i]中求值最小的顶点

k(k,CLOSEST[k])为求得的一条边将顶点k加入到集合U定义:CLOSEST[i]为U中的一个顶点边(i,CLOSEST[i])具有最小的权LOWCOST[i]Prim算法流程图i=2i<=nYesNoVoidPrim(C)详见P135-136CosttypeC[n+1][n+1];{costtypeLOWCOST[n+1];intCLOSEST[n+1];inti,j,k;costtypemin;

for(i=2;i<=n;i++){LOWCOST[i]=C[1][i];CLOSEST[i]=1;}

赋初值

for(i=2;i<=n;i++)

{

min=LOWCOST[i];k=i;for(j=2;j<=n;j++)if(LOWCOST[j]<min){min=LOWCOST[j];k=j;}

求离U中某一顶点最近的顶点

Cout<<“(”<<k<<“,”<<CLOSEST[k]<<“)”<<end1;LOWCOST[k]=INFINITY;将k加入集合U

for(j=2;j<=n;j++)if(C[k][j]<LOWCOST[j]&&LOWCOST[j]!=INFINITY){LOWCOST[j]=C[k][j];CLOSEST[j]=k;}调整

}}CLOSEST[i]为U中的一个顶点边(i,CLOSEST[i])具有最小的权LOWCOST[i]V3V1V4V5V2V66515536462C123456123456∞615∞∞6∞5∞3∞15∞5645∞5∞∞2∞36∞∞6∞∞426∞LOWCOST[i]i=123456123456--615∞-∞-5∞5645∞26∞5∞∞6∞∞∞∞3∞∞∞∞∞∞CLOSEST[i]i=123456123456--111113113331633316333162331622打印边123456K(K,CLOSEST[K])3(3,1)6(6,3)4(4,6)2(2,3)5(5,2)for(j=2;j<=n;j++)if(C[k][j]<LOWCOST[j]&&LOWCOST[j]<INFINITY){LOWCOST[j]=C[k][j];CLOSEST[j]=k;}V3V1V4V5V2V66515536462[例]K=?min(LOWCOST[])CLOSEST[i]为U中的一个顶点边(i,CLOSEST[i])具有最小的权LOWCOST[i]算法要点:

令T=(V,E),(V=1,2,3,…,n),c是关于E中每条边的权函数1、T中每个顶点自身构成一个连通分量;2、按照边的权不减的顺序,依次考查E中的每条边;3、如果被考查的边连接不同的分量中的两个顶点,则合并两个分量;4、如果被考查的边连接同一个分量中的顶点,则放弃,避免环路;5、T中连通分量逐渐减少;当T中的连通分量的个数为1时,说明V中的全部顶点通过E中权最小的那些边,构成了一个没有环路的连通图T,即为最小生成树。2、求最小生成树——Kruskal算法输入:连通图G=(V,E),其中v=(1,2,…,n),C是关于E中的每条弧的权。输出:G的最小生成树VoidKruskal(V,T){T=V;ncomp=n;/*结点个数*/while(ncomp>1){从E中取出删除权最小的边(v,u);if(v和u属于T中不同的连通分量)

{T=T∪{(v,u)}ncomp--;}}}1264537182023122515851067412645371812158564egfhdbca2212121(b)egfhdbca2212121(c)egfhdbca23212412213(a)*4.5无向图的双连通性(Biconnectivity)P1374.4.1无向图的双连通分量设G=(V,E)[定义]

假若在删去顶点v以及和v相关联的边之后,将图的一个连同分量分割成两个或两个以上的连同分量,则称该结点为关节点。abdecfg(a)bdecfg(b)abdefg(c)[定义]

若对V中每个不同的三元组v,w,a;在v和w之间都存在一条不包含a的路,就说G是双连通的(Biconnected)先深搜索和先深编号的作用:

通过是否遇到回退边,即可确定是否有环路。说V中的点a是一个关节点,若V中有结点v和w,使得v,w,a

各不相同且v

和w

之间的每条路都包含a

双连通的无向图是连通的,但连通的无向图未必双连通。一个连通的无向图是双连通的,当且仅当它没有关节点。[定义]

e1=e2

或者有一条环路包含e1又包含

e2

则称边e1

和e2

是等价的。[定义]

设Vi

Ei

中各边所连接的点集(1≤i≤k),每个图

Gi=(Vi,Ei)

叫做

G

的一个双连通分量。双连通分量的性质:

性质1:

Gi

是双连通的(1≤i≤k)

性质2:对所有的i≠j,Vi∩Vj

最多包含一个点

性质3:

v

是G的关节点,当且仅当v∈Vi∩Vj(i≠j)双连通图的研究意义:如通讯网络。上述等价关系将E分成等价类E1,E2,…,Ek,两条不同的边属于同一个类的充要条件是它们在同一个环路上。V1V2V3V4V5V6V9V7V8无向图和它的双连通分量图G图G的双连通分量V1V2V3(1)V4V6(4)V2V4V5(2)V6V9V7V8(3)4.4.2求关节点—对图进行一次先深搜索便可求出所有的关节点若生成树的根有两株或两株以上子树,则此根结点必为关节点

(第一类关节点)。因为图中不存在连接不同子树中顶点的边,因此,若删去根顶点,生成树变成生成森林。若生成树中非叶顶点v,其某株子树的根和子树中的其它结点均没有指向v的祖先的回退边,则v是关节点(第二类关节点)。因为删去v,则其子树和图的其它部分被分割开来1abdecfg234567由先深生成树可得出两类关节点的特性:gcafedb定义low[v]:

设对连通图G=(V,E)进行先深搜索的先深编号

为dfn[v],产生的先深生成树为S=(V,T),B是

回退边之集。对每个顶点v,low[v]定义如下:gcafedbabdecfg12345671111555dfn

lowlow[v]=mindfn[v],dfn[w],low[y](v,w)∈B,w

是顶点v

在先深生成树上有回退边连接的祖先结点;(v,y)∈T,y

是顶点v

在先深生成树上的孩子顶点Low[v]取顶点v和w的深度优先编号的较小者,其中的w是从v点沿着零条或多条树边到v的后代x,之后沿着任意一条回退边(x,w)所能达到的任何顶点算法4.5求无向图的双连通分量输入:连通的无向图G=(V,E)。L[v]表示关于v的邻接表。输出:G的所有双连通分量,每个连通分量由一序列的边组成。算法要点:1.计算先深编号:对图进行先深搜索,计算每个结点v的先深编号

dfn[v],形成先深生成树S=(V,T)。2.计算low[v]:在先深生成树上按后根顺序进行计算每个顶点v的

low[v],low[v]取下述三个结点中的最小者:

(1)dfn[v]:(2)dfn[w]:凡是有回退边(v,w)的任何结点w;

(3)low[y]:对v的任何儿子(树边)y。3.求关节点:

(1)树根是关节点,当且仅当它有两个或两个以上的儿子

(第一类关节点);

(2)非树根结点v是关节点当且仅当v有某个儿子y,使

low[y]≥dfn[v](第二类关节点)。VoidsearchB(v){(1)makev“old”;(2)dfn[v]=count;(3)count++;(4)low[v]=dfn[v];(5)for(eachw∈L[v])(6)if(wismarked”new”)(7){add(v,w)toT;(8)father[w]=v;//w是v的儿子

(9)searchB(w);(10)if(low[w]>=dfn[v])Abiconnectedcomponenthasbeenfound;(11)low[v]=min(low[v],low[w]);}(12)elseif(wisnotfather[v])//(v,w)是回退边

(13)low[v]=min(low[v],dfn[w]);}调用过程:T=¢;count=1;for(allv∈V)makev“new”;searchB(v0);//v0为任意顶点*4.6有向图的搜索DFS和BFS搜索在有向图和无向图中的区别?有向图搜索:树边、向前边、回退边、和横边1、若dfn[v]<dfn[w],则(v,w)是树边或向前边;

此时,visited[v]=“old”,visited[w]=“new”,(v,w)为树边

visited[v]=“old”,visited[w]=“old”,(v,w)为向前边2、若dfn[v]>dfn[w],则(v,w)是回退边或横边;

当产生树边(i,j)时,同时记下j的父亲:father[j]=i,于是对图中任一条边(v,w),当visited[v]=“old”,visited[w]=“old”且dfn[v]>dfn[w]时,由结点v沿着树边向上(father中)查找w(可能直到根);若找到w,则(v,w)是回退边,否则是横边ABECDGHF(a)AECDGHFB12345678图(a)先深生成森林ABECD(b)ABECD12345图(b)先广生成森林*4.7强连通性有向图的强连通分量是满足下列要求的最大子集:对任意两个顶点x

和y

,都存在一条有向路从x

到y

,也存在一条有向路从y

到x。设G=(V,E)是一个有向图,称顶点v,w∈V是等价的,要么v=w;要么从顶点v

到w

有一条有向路,并且从顶点w

到v

也有一条有向路。[定义]设Ei(1≤i≤r)是头、尾均在Vi

中的边集,则Gi

=(Vi,Ei)称为G的一个强连通分量,简称强分量。[定义]上述等价关系将V分成若干个等价类V1,V2,…,Vr强连通图:只有一个强分量的有向图称为强连通图。分支横边:不在任何强连通分支(量)中的边,如:a→d,c→d注:每个结点都是在某个强连通分支中出现,但有些边可能不在任何强分支中。归约图:用强分量代表顶点,用分支横边代表有向边的图称为原图的归约图。显然,归约图是一个不存在环路的有向图,它表示了强分量之间的连通性。adcb(a)adcb(b)a,b,cd归约图adcb(a)eabcd(b)e54321abcd(c)e54321算法:P145,(step1---step4)求强连通图的实例4.8拓扑分类给定一个无环路有向图G=(V,E),各结点的编号为v=(1,2,…,n)。要求对每一个结点i

重新进行编号,使得若i

是j的前导,则有Label[i]<label[j]。即拓扑分类是将无环路有向图排成一个线性序列,使当从结点i

到结点j存在一条边,则在线性序列中,将i排在j

的前面。*+***+e+*+ecdcbb+cdcd*+**+*+ecbcd((a+b)*(b*(c+d))+(c+d)*e)*((c+d)*e)有向无环图及其应用课程代号课程名称先修课代号1计算机原理82编译原理4,53操作系统4,54程序设计无5数据结构4,66离散数学97形式语言68电路基础99高等数学无10计算机网络1例:12435678910v1v2v4v3v6v5v1v2v4v3v5v2v4v3v5v2v3v5v2v5v5可输出结点的入度为零,删去该结点,并将与该顶点相邻的顶点的入度减1。V6V1V4V3V2V5021230入度V1V2V3V4V5v6top=0(a)021231入度(b)021121入度(c)014021入度(d)044011入度(e)044011入度(f)toptoptoptoptopv6→v1→v3→v2→v4→044001入度(f)topv5→044001入度(g)top=0v14v2v3v4v555322^^v654021230^^^^v1v2v4v3v6v561^top链栈的初始状态进栈:ID[i]=top;top=i;退栈:i=top;top=ID[top]021230入度ID[i]123456top=0014021入度top021231入度ID[i]123456top=661^top算法主要步骤:计算每个顶点i的入度ID[i];for(i=1;i<=n;i++)//初始化链栈;

if(ID[i]==0){ID[i]=top;top=i;}for(i=1;i<=n;i++){输出top指针指向的顶点;

k=top;top=ID[top];

顶点k的每一个邻接点j的入度-1if(ID[j]==0){ID[j]=top;top=j;}}进栈:ID[i]=top;top=i;退栈:i=top;top=ID[top]StatusTopologicalsort(ALGRAPHG){FindInDegree(G,indegree);MakeNUlLL(S);for(v=0;v<n;++v)if(!indegree[v])push(v,S);count=0;while(!EMPTY(S)){v=pop(S);printf(v);++count;for(邻接于v的每个顶点w){if(!(--indegree[w]))push(S,w);}}if(count<n)cout<<“图中有环路”;elsereturnOK;}算法一:应用栈StatusTopologicalsort(L){QUEUEQ;MakeNUlLL(Q);for(v=1;v<=n;++v)if(indegree[v]=0)ENQUEUE(v,Q);nodes=0;while(!EMPTY(Q)){v=FRONT(Q);DEQUEUE(Q);cout<<v;nodes++;for(邻接于v的每个顶点w)if(!(--indegree[w]))ENQUEUE(w,Q);}if(nodes<n)cout<<“图中有环路”;}算法二:应用队列算法三:DFS遍历生成拓扑序列Voidtopodfs(v){PUSH(v,S);mark[v]=TRUE;for(L[v]中的每一个顶点w)doif(mark[w]=FALSE)topodfs(w);printf(top(S));POP(S);}Voiddfs-topo(GRAPHL){MAKENULL(S);for(u=1;u<=n;u++)make[u]=FALSE;for(u=1;u<=n;u++)if(!mark[u])topodfs(u);}思想:借助栈,在DFS中,把第一次遇到的顶点入栈,到达某一顶点递归返回,从栈中弹出顶点并输出。AOE网(ActivityOnEdgeNetwork)

在带权的有向图中,用结点表示事件,用边表示活动,边上权表示活动的开销(如持续时间),则称此有向图为边表示活动的网络,简称AOE网。4.9关键路径v7源点起始点v1v4v3v2v5v6v9a1=6a6=2a3=5a2=4a5=1a4=1a7=9a8=7a9=4a11=4a10=2汇点结束点AOE网v8AOE网的性质只有在某个顶点所代表的事件发生后,从该顶点出发的各有向边代表的活动才能开始;只有在进入某一顶点的各有向边代表的活动已经结束,该顶点所代表的事件才能发生;表示实际工程计划的AOE网应该是无环的,并且存在唯一的入度为0的开始顶点(源点)和唯一的出度为0的结束点(汇点)。v7源点起始点v1v4v3v2v5v6v9a1=6a6=2a3=5a2=4a5=1a4=1a7=9a8=7a9=4a11=4a10=2汇点结束点AOE网v8AOE网研究的主要问题:如果用AOE网表示一项工程,那么仅仅考虑各个子工程之间的优先关系还不够,更多地是关心整个工程完成的最短时间是多少,哪些活动的延迟将影响整个工程进度,而加速这些活动能否提高整个工程的效率,因此AOE网有待研究的问题是:(1)完成整个工程至少需要多少时间?(2)哪些活动是影响工程进度的关键活动?v7源点起始点v1v4v3v2v5v6v9a1=6a6=2a3=5a2=4a5=1a4=1a7=9a8=7a9=4a11=4a10=2汇点结束点AOE网v8路径长度、关键路径、关键活动:路径长度:是指从源点到汇点路径上所有活动的持续时间之和。关键路径:在AOE网中,由于有些活动可以并行,所以完成工程的最短时间是从源点到汇点的最大路径长度。因此,把从源点到汇点具有最大长度的路径称为关键路径。一个AOE中,关键路径可能不只一条。关键活动:关键路径上的活动称为关键活动。v7源点起始点v1v4v3v2v5v6v9a1=6a6=2a3=5a2=4a5=1a4=1a7=9a8=7a9=4a11=4a10=2汇点结束点AOE网v8①事件Vj

的最早可能发生时间VE(j)

是从源点V1到顶点Vj的最长路径长度。②活动ai的最早可能开始时间E(k)

设活动ai在边<Vj,Vk>上,则E(i)也是从源点V1到顶点Vj

的最长路径长度。这是因为事件Vj发生表明以Vj为起点的所有活动ai可以立即开始。因此,

E(i)=VE(j)…………..(1)关键路径和关键活动性质分析:(与计算关键活动有关的量)v7源点起始点v1v4v3v2v5v6v9a1=6a6=2a3=5a2=4a5=1a4=1a7=9a8=7a9=4a11=4a10=2汇点结束点AOE网v8③事件Vk的最迟发生时间VL(k)

是在保证汇点Vn在VE(n)时刻完成的前提下,事件Vk的允许的最迟开始时间。在不推迟工期的情况下,一个事件最迟发生时间VL(k)应该等于汇点的最早发生时间VE(n)减去从Vk到Vn的最大路径长度。v7源点起始点v1v4v3v2v5v6v9a1=6a6=2a3=5a2=4a5=1a4=1a7=9a8=7a9=4a11=4a10=2汇点结束点AOE网v8④活动ai的最迟允许开始时间L(i)

是指在不会引起工期延误的前提下,活动ai允许的最迟开始时间。因为事件Vk发生表明以Vk为终点的入边所表示的所有活动均已完成,所以事件Vk的最迟发生时间VL(k)也是所有以Vk为终点的入边<Vj,Vk>所表示的活动ai可以最迟完成时间。显然,为不推迟工期,活动ai的最迟开始时间L(i)应该是ai的最迟完成时间VL(k)减去ai的持续时间,即

L(i)=VL(k)-ACT[j][k]…………..(2)

其中,ACT[j][k]是活动ai的持续时间(<Vj,Vk>上的权)。VjVkai⑤时间余量L(i)-E(i)

L(i)-E(i)表示活动ak的最早可能开始时间和最迟允许开始时间的时间余量。关键路径上的活动都满足:L(i)=E(i)…………..(3)

L(i)=E(i)表示活动是没有时间余量的关键活动。

由上述分析知,为找出关键活动,需要求各个活动的E(i)与L(i),以判别一个活动ai是否满足L(i)=E(i)。E(i)和L(i)可有公式(1)和(2)。而VE(k)

和VL(k)可由拓扑分类算法得到。利用拓扑分类算法求关键路径和关键活动。Step1(前进阶段):从源点V1出发,令VE(1)=0,按拓扑序列次序求出其余各顶点事件的最早发生时间:利用拓扑分类算法求关键路径和关键活动

其中T是以顶点Vk为尾的所有边的头顶点的集合(2≤k≤n)如果网中有回路,不能求出关键路径则算法中止;否则转Step2。其中S是以顶点Vj为头的所有边的尾顶点的集合(2≤j≤n-1)j∈TVE(k)=max{VE(j)+ACT[j][k]

}Step2(回退阶段):从汇点Vn出发,令VL(n)=VE(n),按逆拓扑有序求其余各顶点的最晚发生时间:k∈SVL(j)=min{VL(k)+ACT[j][k]}Step3:求每一项活动ai的最早开始时间:

E(i)=VE(j)

最晚开始时间:

L(i)=VL(k)-ACT[j][k]

若某条边满足E(i)=L(i),则它是关键活动。为了简化算法,可以在求关键路径之前已经对各顶点实现拓扑排序,并按拓扑有序的顺序对各顶点重新进行了编号。不是任意一个关键活动的加速一定能使整个工程提前。想使整个工程提前,要考虑各个关键路径上所有关键活动。例1:活动e(i)l(i)l(i)-e(i)a1a2a3a4a5a6a7a8a9a10a1100002203366046258377077071031616014140注:Ve(i):事件最早可能发生时间源点到达该事件的最长路经Vl(i):事件最迟发生时间

VE(n)-maxLik

e(i):活动最早可能开始时间

Ve(活动起点)

l(i):活动最迟允许开始时间

Vl(活动终点)-ai

事件vevlv1v2v3v4v5v6v7v8v90066465877710161614141818v7源点起始点v1v4v3v2v5v6v9a1=6a6=2a3=5a2=4a5=1a4=1a7=9a8=7a9=4a11=4a10=2汇点结束点AOE网v8123654a1=3a2=2a4=3a3=2a5=4a6=1a7=2a8=3例2:顶点Ve(i)Vl(i)活动e(i)l(i)l(i)-e(i)123456a1a2a3a4a5a6a7a8032668042678003326621044276510110103注:事件最早可能发生时间Ve(i)

源点到达该事件的最长路经事件最迟发生时间Vl(i)VE(n)-maxLik活动最早可能开始时间e(i)Ve(活动起点)活动最迟允许开始时间l(i)

Vl(活动终点)-ai

4.10单源最短路径D[1]D[2]D[3]D[4]D[5]∞10∞30100D∞10∞30100∞∞50∞∞∞∞∞2010∞∞20∞60∞∞∞∞∞C集合S={1}{1,2,3,4,5}20124351010030105060V0Dijkstra算法基本思想:集合S的初值为S={1}D为各顶点当前最小路径从V-S中选择顶点w,使D[w]的值最小并将w加入集合S,则w的最短路径已求出。调整其他各结点的当前最小路径

D[k]=min{D[k],D[w]+C[w][k]}直到S中包含所有顶点12435101003010502060V0循环SwD[2]D[3]D[4]D[5]初态{1}-10∞301001{1,2}21060301002{1,2,4}4105030903{1,2,4,3}3105030604{1,2,4,3,5}510503060算法的逐步求精过程:算法:VoidDijkstra(G){S={1};for(i=2;i<=n;i++)D[i]=C[1][i];for(i=1;i<n;i++){从V-S中选出一个顶点w,使D[w]最小;

S=S+{w};for(V-S中的每一个顶点v)D[v]=min(D[v],D[w]+C[w][v]);}}算法:VoidDijkstra(GRAPHG,costtypeD[MAXVEX+1]){intS[MAXVEX+1];for(i=1;i<=n;i++){D[i]=G[1][i];S[i]=FALSE;}S[1]=TRUE;for(i=1;i<n;i++){w=mincost(D,S);S[w]=TRUE;for(v=2;v<=n;n++)if(S[v]!=TRUE){sum=D[w]+G[w][v];if(sum<D[v])D[v]=sum;}}}intmincost(D,S){temp=INFINITY;w=2;for(i=2;i<=n;i++)if(!S[i]&&D[i]<temp){temp=D[i];w=i;}returnw;}最小路径经过哪些点?算法:VoidDijkstra(GRAPHG,costtypeD[MAXVEX+1]){intS[MAXVEX+1],P[MAXVEX+1];for(i=1;i<=n;i++){D[i]=G[1][i];S[i]=FALSE;P[i]=1;}S[1]=TRUE;for(i=1;i<n;i++){w=mincost(D,S);S[w]=TRUE;for(v=2;v<=n;n++)if(S[v]!=TRUE){sum=D[w]+G[w][v];if(sum<D[v]){D[v]=sum;p[v]=w;

}}}}P[1]P[2]P[3]P[4]P[5]11413P循环SwD[2]D[3]D[4]D[5]D[6]初态{1}-∞

10∞301001{1,3}3∞106030

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论