版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
第8章图总体要求:熟悉图的定义熟悉图的抽象数据类型描述中各种操作的含义掌握图的存储结构熟练掌握图各种存储接结构下的建立、遍历算法熟练掌握带权图的最小生成树的定义和普里姆算法、克鲁斯卡尔算法熟练掌握带权图的最短路径的定义和迪杰斯特拉算法、弗洛伊德算法核心技能点:具有图理论应用于实际问题的能力1第8章图8.1图的实例及概念8.1.1实例8.1.2图的定义和基本概念8.2图的存储结构及实现8.2.1邻接矩阵8.2.2邻接链表8.2.3图的ADT定义8.3遍历图8.3.1深度优先搜索法8.3.2广度优先搜索2第8章图8.4最小生成树8.4.1最小生成树的基本概念8.4.2普里姆算法8.4.3克鲁斯卡尔算法8.5最短路径8.5.1从某个源点到其它各顶点的最短路径8.5.2求每一对顶点之间最短路径8.6上机实验本章小结习题3第8章图扩展技能点:图各种存储结构和算法C语言环境下的实现相关知识点:C语言数组的知识C语言结构体的知识C语言指针的知识C语言函数的知识离散数学图论的知识4第8章图学习重点:熟悉图的定义熟悉图的抽象数据类型描述中各种操作的含义掌握图的存储结构熟练掌握图各种存储接结构下的建立、遍历算法熟练掌握带权图的最小生成树的定义和普里姆算法、克鲁斯卡尔算法熟练掌握带权图的最短路径的定义和迪杰斯特拉算法、弗洛伊德算法5第8章图图(Graph)是一种复杂的非线性结构。在线性表中,数据元素满足唯一的线性关系,每个元素(除第一个元素和最后一个元素之外),只有一个直接前趋和直接后继;在树型结构中,数据元素有明显的层次关系,并且每个元素只与上层一个元素(双亲结点)及下层中多个元素(孩子结点)相关;而在图型结构中,数据元素之间的关系是任意的,任何两个元素都可以相关,因此它较线性结构和树结构更复杂。图在各个领域的应用是十分广泛的。在计算机的应用领域如开关理论、逻辑设计、人工智能、形式语言、操作系统、编译程序以及信息检索内,图都起着重要的作用;在其他领域如电路分析、项目规划、遗传学、控制论以及一些社会科学中,图的应用也非常普遍。有关图论的内容是离散数学的主要内容之一,因此本章将主要讨论图在计算机中的存储表示、操作的实现及典型的应用等。6第8章图8.1图的实例及概念8.1.1实例在日常生活中,对象和对象之间的关系常常可以用包括点和线的示意图来表示。如图8.1表示的是我国部分省市之间的铁路交通示意图,反映了这几个城市之间的铁路分布情况。其中,省市用点表示,点和点之间的连线代表了对应的两个省市之间的铁路线。7第8章图8
从以上例子可以看出,点及点之间的连线可被用来反映客观世界中某些对象之间的特定关系。图8.1我国部分省市之间的铁路交通示意图第8章图8.1.2图的定义和基本概念1.图的定义图(graph)是由顶点集合(vertex)及顶点间的关系集合组成的一种数据结构:Graph=(V,E)其中,V={x|x∈某个数据对象}是顶点的有穷非空集合;E={(x,y)|x,y∈V}或E={<x,y>|x,y∈V&&Path(x,y)}是顶点之间关系的有穷集合,也叫做边(Edge)集合。Path(x,y)表示从x到y的一条单向通路,它是有方向的。9第8章图2.一些与图有关的基本概念(1)有向图(DirectedGraph)与无向图(UndirectedGraph)。在有向图中,顶点对<x,y>是有序的,它称为从顶点x到顶点y的一条有向边。因此,<x,y>与<y,x>是不同的两条边。此时,顶点对<x,y>用一对尖括号括起来,x是有向边的始点,y是有向边的终点。在无向图中,顶点对(x,y)是无序的,它称为与顶点x和顶点y相关联的一条边,这条边没有特定的方向,(x,y)与(y,x)是同一条边。一般为了有别于有向图,顶点对用一对圆括号括起来。图8.2中有4个图,其中图G1和G2是无向图,G1的顶点集合为V(G1)={0,l,2,3},边集合为E(G1)={(0,1),(0,2),(0,3),(1,2),(1,3),(2,3)}。G2的顶点集合为V(G2)={0,1,2,3,4,5,6},边集合为E(G2)={(0,1),(0,2),(1,3),(1,4),(2,5),(2,6)}。在无向图中边上不加箭头。图G3和G4是有向图,G3的顶点集合为V(G3)={0,1,2},边集合为E(G3)={<0,1>,<l,0>,<1,2>}。G4的顶点集合为V(G4)={0,1,2,3},边集合为E(G4)={<0,1>,<l,0>,<0,2>,<2,0>,<0,3>,<3,0>,<l,2>,<2,1>,<1,3>,<3,1>,<2,3>,<3,2>}。在有向图中,为了表示有向边,按边的方向用箭头画出,箭头从边的始点指向终点。10第8章图11图8.2图的示例第8章图并不是所有的图都是我们的研究对象,在讨论图的时候,对讨论的对象有一些限制:①图中不能有从自身到自身的边(即自身环SelfLoop),就是说不应有形如(x,x)或<y,y>的边。如图8.3(a)的带自身环的图不在本章讨论之内。②与两个特定顶点相关联的边不能多于一条。如图8.3(b)所示的多重图也不讨论。12图8.3本章不予讨论的图(a)带自身环的图(b)多重图第8章图(2)完全图(CompleteGraph)。在有n个顶点的无向图中,若有n(n-1)/2条边,则称此图为完全无向图,如图8.2(a)所示的图G1。在有n个顶点的有向图中,若有n(n-1)条边,则称此图为完全有向图,如图8.1(d)所示的图G4。完全图中的边数达到最大。(3)权(Weight)。有些图的边附带有数据信息,这些附带的数据信息称为权。权可以表示实际问题中从一个顶点到另一个顶点的距离、花费的代价、所需的时间,等等。带权的图称作网络(Network)。图8.4就是带权图。其中,图8.4(a)是一个工程的施工进度图,图8.4(b)是一个交通网络图。13第8章图14图8.4带权的图(a)是一个工程的施工进度图(b)是一个交通网络图第8章图(4)邻接顶点(AdjacentVertex)。如果(x,y)是E中的一条边,则称x与y互为邻接顶点,且边(x,y)依附于顶点x和y。在图8.2的G2中,与顶点1相邻接的顶点有0,3,4。在G2中依附于顶点2的边有(0,2)、(2,5)和(2,6)。如果<x,y>是图中的一条有向边,则称顶点x邻接到顶点y,顶点y邻接自顶点x,边<x,y>是顶点x的出边(Outedge),是顶点y的入边(Inedge)。例如,在图8.2的G3中,有向边<1,2>的顶点1邻接到顶点2,顶点2邻接自顶点1,顶点1的出边是<1,0>和<1,2>,入边是<0,1>。(5)子图(Subgraph)。设有两个图G1=x(V,E)和G2=(V,E)。若V(G1)V(G2)且E(G1)E(G2)则称图G1是图G2的子图。15第8章图(6)顶点的度(Degree)。一个顶点v的度是与它相关联的边的条数,记作TD(v)。在有向图中,顶点的度等于该顶点的人度与出度之和。其中,顶点v的人度是以v为终点的有向边(入边)的条数,记作ID(v);顶点v的出度是以v为始点的有向边(出边)的条数,记作OD(v)。顶点v的度TD(v)=ID(v)+OD(v)。一般地,若图G中有n个顶点,e条边(或弧),则
E={{}(7)路径(Path)。在图G=<V,E)中,若从顶点vi出发,沿一些边经过一些顶点vp1,vp2,…vpm到达顶点vj。则称顶点序列(vi,vp1,vp2,…vpm,vj)为从顶点vi到顶点vj的路径。它经过的边(vi,vp1),(vp1,vp2),…(vpm,vj)应是属于E的边。16第8章图(7)路径(Path)。在图G=<V,E)中,若从顶点vi出发,沿一些边经过一些顶点vp1,vp2,…vpm到达顶点vj。则称顶点序列(vi,vp1,vp2,…vpm,vj)为从顶点vi到顶点vj的路径。它经过的边(vi,vp1),(vp1,vp2),…(vpm,vj)应是属于E的边。(8)路径长度(PathLength)。对于不带权的图,路径长度是指此路径上边的条数。对于带权图,路径长度是指路径上各边的权之和。(9)简单路径与回路(Cycle)。若路径上各顶点v1,v2…vmv1,均不互相重复,则称这样的路径为简单路径。若路径上第一个顶点v1与最后一个顶点vm重合,则称这样的路径为回路或环。如图8.5所示。图8.5(a)所示是简单路径,图8.5(b)所示是非简单路径。通常在解决实际应用问题时,只考虑简单路径。图8.5(c)所示是回路。17第8章图18图8.5简单路径与回路(a)简单路径(b)非简单路径(c)回路第8章图(10)连通图与连通分量(ConnectedGraph&ConnectedComponent)。在无向图中若从顶点v1到顶点v2有路径,则称顶点v1与v2是连通的。如果图中任意一对顶点都是连通的,则称此图是连通图。非连通图的极大连通子图叫做连通分量。(11)强连通图与强连通分量(StronglyConnectedGraph)。在有向图中,若对于每一对顶点vi和vj,都存在一条从vi到vj和从vj到vi的路径,则称此图是强连通图。非强连通图的极大强连通子图叫做强连通分量。(12)生成树(SpanningTree)。一个连通图的生成树是它的极小连通子图,在n个顶点的情形下,有n-l条边。但有向图则可能得到它的由若干有向树组成的生成森林。19第8章图8.2图的存储结构及实现图的存储结构比较多。对于图的存储结构的选择取决于具体的应用和需要进行的运算。下面给出两种常用的图的结构,邻接矩阵和邻接表。8.2.1邻接矩阵在图的邻接矩阵(AdjacencyMatrix)表示中,除了一个记录各个顶点信息的顶点表外,还有一个称之为邻接矩阵的表示各个顶点之间关系的矩阵。若设图A=(V,E)是一个有n个顶点的图,则图的邻接矩阵是一个二维数组A.edge[n][n],它的定义为:1,if((i,j)or<i,j>∈E)A.edge[i][j]=0,otherwise20第8章图图8.6给出了无向图和它的邻接矩阵,图8.7给出了有向图和它的邻接矩阵。从图8.6中可知,无向图的邻接矩阵是对称的,将第i行的元素值或第i列的元素值累加起来就得到顶点i的度。但有向图的邻接矩阵可能是不对称的。如果第i行的A.Edge[i][j]=1,则表示有一条从顶点i到顶点j的有向边,将第i行的所有元素值累加起来就得到顶点i的出度;如果第j列的A.Edge[k][j]=l,则表示有一条从顶点k到顶点j的有向边,将第j列的所有元素值累加起来就得到顶点j的入度。21第8章图22图8.6无向图和它的邻接矩阵图8.7有向图和它的邻接矩阵第8章图对于网络(或带权图),邻接矩阵定义如下:
Wi,j,if((i!=j)&&<i,j>∈Eor(i,j)∈E)A.edge[i][j]=∝,otherwise,but(i≠j)0,if(i==j)图8.8给出一个网络的邻接矩阵。23图8.8一个网络和它的邻接矩阵第8章图对于用邻接矩阵表示的图来说,要想确定“图中有多少条边?”和“图是否连通?”等问题,需要检查除对角线外的所有n2-n个矩阵元素,时间开销很高。当图中的边数e远远小于图的邻结矩阵元素个数n2时,图的邻接矩阵变成稀疏矩阵,存储利用率很低。为此,可以改用后面介绍的邻接表。下面给出用邻接矩阵作为存储表示的图的类型定义及算法实现。其中有一个类型为顺序表的顶点表向量Vertices,用以存储顶点的信息,还有—个作为邻接矩阵使用的二维数组edge,用以存储图中的边,其矩阵元素个数取决于顶点个数,与边数无关。当前顶点个数在顺序表中声明,当前边数由CurrentEdges指明。24第8章图以下的类型定义和函数都放在邻接矩阵头文件AdjMGraph.h中。#include"SeqList.h"/*包含顺序表头文件*/typedefstruct{SeqListVertices;/*存放顶点的顺序表*/intedge[MaxVertices][MaxVertices];/*存放边的邻接矩阵*/intCurrentEdges;/*边的条数*/}AdjMWGraph;/*图的结构体定义*/1.邻接矩阵的初始化邻接矩阵的初始化可表示为:voidInitiate(AdjMWGraph*G,intn);其中参数G表示指定的邻接矩阵,其类型为AdjMWGraph,参数n表示顶点个数。函数的功能:实现邻接矩阵的初始化。25第8章图处理过程为:⑴使图G邻接矩阵的对角线元素为0,其它元素为∞;⑵使图G邻接矩阵的边数为0;⑶图G顶点表向量Vertices的初始化。函数实现如下:voidInitiate(AdjMWGraph*G,intn){inti,j;for(i=0;i<n;i++)for(j=0;j<n;j++){if(i==j)G->edge[i][j]=0;elseG->edge[i][j]=MaxWeight;}26第8章图G->CurrentEdges=0;/*边的条数置为0*/ListInitiate(&G->Vertices);/*顺序表初始化*/}2.图的邻接矩阵加入新顶点图的邻接矩阵加入新顶点可表示为:voidInsertVertex(AdjMWGraph*G,DataTypevertex);其中参数G表示指定的邻接矩阵,其类型为AdjMWGraph,参数vertex表示顶点,其类型为DataType。函数的功能:实现图的邻接矩阵加入新顶点。27第8章图处理过程为:⑴图G顶点表向量Vertices中增加一个新元素,元素个数加1,即在顺序表表尾插入;⑵返回。函数实现如下:voidInsertVertex(AdjMWGraph*G,DataTypevertex)/*在图G中插入顶点vertex*/{ListInsert(&G->Vertices,G->Vertices.size,vertex);}3.图的邻接矩阵加入新边图的邻接矩阵加入新边可表示为:voidInsertEdge(AdjMWGraph*G,intv1,intv2,intweight);其中参数G表示指定的邻接矩阵,其类型为AdjMWGraph,参数v1,v2表示顶点编号,其类型为int,weight为边的权值。28第8章图函数的功能:实现图的邻接矩阵加入新边。处理过程为:若v1,v2合法,则⑴使图G->edge[v1][v2]=weight;⑵G->CurrentEdges++。函数实现如下:voidInsertEdge(AdjMWGraph*G,intv1,intv2,intweight)/*在图G中插入边<v1,v2>,边<v1,v2>的权为weight*/{if(v1<0||v1>G->Vertices.size||v2<0||v2>G->Vertices.size){printf("参数v1或v2越界出错!\n");exit(1);}G->edge[v1][v2]=weight;G->CurrentEdges++;}29第8章图4.图的邻接矩阵删除边图的邻接矩阵删除边可表示为:voidDeleteEdge(AdjMWGraph*G,intv1,intv2);其中参数G表示指定的邻接矩阵,其类型为AdjMWGraph,参数v1,v2表示顶点编号,其类型为int。函数的功能:实现图的邻接矩阵删除边。处理过程为:若v1,v2合法,则⑴使图G->edge[v1][v2]=MaxWeight;⑵G->CurrentEdges--。30第8章图函数实现如下:voidDeleteEdge(AdjMWGraph*G,intv1,intv2)/*在图G中删除边<v1,v2>*/{if(v1<0||v1>G->Vertices.size||v2<0||v2>G->Vertices.size||v1==v2){printf("参数v1或v2越界出错!\n");exit(1);}G->edge[v1][v2]=MaxWeight;G->CurrentEdges--;}31第8章图4.图的邻接矩阵删除顶点图的邻接矩阵删除顶点可表示为:voidDeleteVertex(AdjMWGraph*G,intv);其中参数G表示指定的邻接矩阵,其类型为AdjMWGraph,参数v表示顶点编号,其类型为int。函数的功能:实现图的邻接矩阵删除顶点。处理过程为:⑴若G->edge[i][j]是与该顶点相关的所有边,G->CurrentEdges--;⑵删除该顶点所在的行;⑶删除该顶点所在的列;⑷图G顶点表向量Vertices中删除该元素,元素个数减1。32第8章图函数实现如下:voidDeleteVertex(AdjMWGraph*G,intv)/*删除图G顶点v*/{intn=ListLength(G->Vertices),i,j;DataTypex;for(i=0;i<n;i++)for(j=0;j<n;j++)if((i==v||j==v)&&G->edge[i][j]>0&&G->edge[i][j]<MaxWeight)G->CurrentEdges--;/*被删除边计数*/33第8章图for(i=v;i<n;i++)for(j=0;j<n;j++)G->edge[i][j]=G->edge[i+1][j];/*删除第v行*/for(i=0;i<n;i++)for(j=v;j<n;j++)G->edge[i][j]=G->edge[i][j+1];/*删除第v列*/ListDelete(&G->Vertices,v,&x);/*删除结点v*/}34第8章图5.寻找第一个邻接顶点寻找第一个邻接顶点可表示为:intGetFirstVex(AdjMWGraphG,intv);其中参数G表示指定的邻接矩阵,其类型为AdjMWGraph,参数v表示顶点编号,其类型为int。函数的功能:在图G中寻找序号为v的顶点的第一个邻接顶点,如果这样的邻接顶点存在则返回该邻接顶点的序号,否则返回-1。处理过程为:若v合法,在邻接矩阵的第v行,寻找G.edge[v][col]>0&&G.edge[v][col]<MaxWeight)的元素。寻找到,返回col,否则返回-1。35第8章图函数实现如下:intGetFirstVex(AdjMWGraphG,intv)/*在图G中寻找序号为v的顶点的第一个邻接顶点*//*如果这样的邻接顶点存在则返回该邻接顶点的序号,否则返回-1*/{intcol;if(v<0||v>=G.Vertices.size){printf("参数v1越界出错!\n");exit(1);}for(col=0;col<=G.Vertices.size;col++)if(G.edge[v][col]>0&&G.edge[v][col]<MaxWeight)returncol;return-1;}36第8章图6.寻找下一个邻接顶点寻找下一个邻接顶点可表示为:intGetNextVex(AdjMWGraphG,intv1,intv2);其中参数G表示指定的邻接矩阵,其类型为AdjMWGraph,参数v1,v2表示顶点编号,其类型为int。函数的功能:在图G中寻找序号为v1顶点的邻接顶点v2的下一个邻接顶点,如果这样的邻接顶点存在则返回该邻接顶点的序号,否则返回-1。处理过程为:若v1,v2合法,在邻接矩阵的第v1行,从v2+1开始寻找G.edge[v1][col]>0&&G.edge[v1][col]<MaxWeight)的元素。寻找到,返回col,否则返回-1。37第8章图函数实现如下:intGetNextVex(AdjMWGraphG,intv1,intv2)/*在图G中寻找v1顶点的邻接顶点v2的下一个邻接顶点*//*如果这样的邻接顶点存在则返回该邻接顶点的序号,否则返回-1*//*这里v1和v2都是相应顶点的序号*/{intcol;if(v1<0||v1>=G.Vertices.size||v2<0||v2>=G.Vertices.size){printf("参数v1或v2越界出错!\n");exit(1);}for(col=v2+1;col<G.Vertices.size;col++)if(G.edge[v1][col]>0&&G.edge[v1][col]<MaxWeight)returncol;return-1;}38第8章图7.建立邻接矩阵建立邻接矩阵可表示为:voidCreatGraph(AdjMWGraph*G,DataTypeV[],intn,RowColWeightE[],inte);其中参数G表示指定的邻接矩阵,其类型为AdjMWGraph,参数V表示顶点数组,其类型为DataType,参数n表示顶点数,其类型为int,参数E表示边数组,其类型为RowColWeight,函数外定义,参数e表示边数,其类型为int。函数的功能:在图G中插入n个顶点信息V和e条边信息。处理过程为:⑴顶点顺序表初始化⑵顶点插入⑶边插入39第8章图函数实现如下:/*建立邻接矩阵的AdjMGraphCreate.h*/typedefstruct{introw;/*行下标*/intcol;/*列下标*/intweight;/*权值*/}RowColWeight;/*边信息三元组结构体定义*/voidCreatGraph(AdjMWGraph*G,DataTypeV[],intn,RowColWeightE[],inte)/*在图G中插入n个顶点信息V和e条边信息E*/{inti,k;Initiate(G,n);/*顶点顺序表初始化*/for(i=0;i<n;i++)InsertVertex(G,V[i]);/*顶点插入*/for(k=0;k<e;k++)InsertEdge(G,E[k].row,E[k].col,E[k].weight);/*边插入*/}40第8章图有兴趣的读者,可以用下面函数测试,看一看执行的结果。/*测试主函数*/#include<stdio.h>#include<stdlib.h>typedefcharDataType;#defineMaxSize100#defineMaxVertices10#defineMaxEdges100#defineMaxWeight10000#include"AdjMGraph.h"#include"AdjMGraphCreate.h"41第8章图voidmain(void){AdjMWGraphg1;DataTypea[]={'A','B','C','D','E'};RowColWeightrcw[]={{0,1,10},{0,4,20},{1,3,30},{2,1,40},{3,2,50}};intn=5,e=5;inti,j;CreatGraph(&g1,a,n,rcw,e);DeleteEdge(&g1,0,4);/*删除边<0,4>*/DeleteVertex(&g1,3);/*删除顶点3*/42第8章图printf("顶点集合为:");for(i=0;i<g1.Vertices.size;i++)printf("%c",g1.Vertices.list[i]);printf("\n");printf("权值集合为:\n");for(i=0;i<g1.Vertices.size;i++){for(j=0;j<g1.Vertices.size;j++)printf("%5d",g1.edge[i][j]);printf("\n");}}43第8章图8.2.2邻接链表图的邻接链表(AdjacencyList)存储结构是一种顺序分配和链式分配相结合的存储结构。它包括两个部分,一部分是向量;另一部分是链表。原则上,在链表部分共有n个链表,即每个顶点一个链表。每个链表由一个表头结点和若干个表结点组成。表头结点用来指示第i个顶点vi的数据信息(data)、是否被访问过(tag)和所对应的链表指针(adj);表结点由顶点编号域(vertexnum)和链域(next)所组成,对带权图还可增加权值域(weight)。顶点域指示了与vi相邻接的顶点的序号,所以一个表结点实际代表了一条依附于vi的边;链域指示了依附于vi的另一条边的表结点。因此,第i个链表就表示了依附于vi的所有边。对有向图来讲,第i个链表就表示了从vi发出的所有弧。如果要表示指向vi的弧,则要采用逆邻接链表,第i个链表就表示了指向vi的所有弧。邻接链表中的表头部分是向量,用来存储n个表头结点。向量的下标指示了顶点的序号。44第8章图在C语言中,图的邻接表存储表示如下。typedefstructNode{intvertexNum;/*顶点编号域*/intweight/*权值域*/structNode*next;/*链表指针*/}Edge;/*表结点*/typedefstruct{DataTypedata;/*结点信息*/inttag;/*标识域*/Edge*adj;/*头指针*/}AdjLHeight;/*表头结点45第8章图typedefstruct{AdjLHeighta[MaxVertices];/*表头向量*/intnumOfVerts;/*顶点数*/intnumOfEdges;/*边数*/}AdjLGraph;例如,图8.9为无向图及其邻接链表,图8.10为有向图及其邻接链表、逆接链表,图8.11为有向带权图和其邻接链表。46第8章图47图8.9无向图和其邻接链表(a)无向图(b)邻接链表第8章图48图8.10有向图及其邻接链表、逆接链表(a)无向图(b)邻接链表(c)逆接链表第8章图49图8.11有向带权图和其邻接链表(a)有向带权图(b)邻接链表第8章图下面给出用邻接链表作为存储表示的图的类型定义及常用算法实现。以上的类型定义和函数都放在邻接表头文件AdjLGraph.h中。1.邻接表的初始化邻接表的初始化可表示为:AdjInitiate(AdjLGraph*G);其中参数G表示指定的邻接表,其类型为AdjLGraph。函数的功能:实现邻接表的初始化。处理过程为:⑴使图G邻接表顶点数为0;⑵使图G邻接表的边数为0;⑶图G顶点表向量a初始化。50第8章图函数实现如下:AdjInitiate(AdjLGraph*G)/*图的邻接表初始化*/{inti;G->numOfVerts=0;G->numOfEdges=0;for(i=0;i<MaxVertices;i++){G->a[i].tag=0;G->a[i].adj=NULL;}}51第8章图2.销毁邻接表销毁邻接表可表示为:AdjDestroy(AdjLGraph*G);其中参数G表示指定的邻接表,其类型为AdjLGraph。函数的功能:实现邻接表的销毁。处理过程为:从0号顶点开始,释放图G邻接表中顶点。函数实现如下:52第8章图AdjDestroy(AdjLGraph*G)/*销毁邻接表*/{inti;Edge*p,*q;for(i=0;i<G->numOfVerts;i++){p=G->a[i].adj;while(p!=NULL){q=p->next;free(p);p=q;}}}53第8章图3.图的邻接表加入新顶点图的邻接表加入新顶点可表示为:voidInsertVertex(AdjLGraph*G,inti,DataTypevertex);其中参数G表示指定的邻接表,其类型为AdjLGraph,参数vertex表示顶点,其类型为DataType。I为顶点编号。函数的功能:实现图的邻接表加入新顶点。处理过程为:⑴若i合法,图G顶点表向量a[i]中增加一个新元素,顶点个数加1;⑵返回。54第8章图函数实现如下:voidInsertVertex(AdjLGraph*G,inti,DataTypevertex)/*邻接表插入顶点*/{if(i>=0&&i<MaxVertices){G->a[i].data=vertex;G->numOfVerts++;}elseprintf("顶点越界");}55第8章图4.图的邻接表加入新边图的邻接表加入新边可表示为:voidInsertEdge(AdjLGraph*G,intv1,intv2);其中参数G表示指定的邻接表,其类型为AdjLGraph,参数v1,v2表示顶点编号,其类型为int。函数的功能:实现图的邻接表加入新边。处理过程为:若v1,v2合法,则⑴建立新结点p,p->next=G->a[v1].adj,G->a[v1].adj=p;⑵G->numOfEdges++。56第8章图函数实现如下:voidInsertEdge(AdjLGraph*G,intv1,intv2)/*邻接表插入边*/{Edge*p;if(v1<0||v1>=G->numOfVerts||v2<0||v2>=G->numOfVerts){printf("参数v1或v2越界出错!");exit(0);}p=(Edge*)malloc(sizeof(Edge));p->next=G->a[v1].adj;G->a[v1].adj=p;G->numOfEdges++;}57第8章图5.图的邻接表删除边图的邻接矩阵删除边可表示为:voidDeleteEdge(AdjLGraph*G,intv1,intv2);其中参数G表示指定的邻接表,其类型为AdjLGraph,参数v1,v2表示顶点编号,其类型为int。函数的功能:实现图的邻接表删除边。处理过程为:若v1,v2合法,则⑴查找v2在v1邻接结点中,若存在,则删除到⑵,否则,返回;⑵G->numOfEdges--。58第8章图函数实现如下:voidDeleteEdge(AdjLGraph*G,intv1,intv2)/*邻接表删除边*/{Edge*curr,*pre;if(v1<0||v1>=G->numOfVerts||v2<0||v2>=G->numOfVerts){printf("参数v1或v2越界出错!");exit(0);}pre=NULL;curr=G->a[v1].adj;while(curr!=NULL&&curr->vertexNum!=v2){pre=curr;curr=curr->next;}59第8章图if(curr!=NULL&&curr->vertexNum==v2&&pre==NULL){/*v1的第一邻接点是v2*/G->a[v1].adj=curr->next;free(curr);G->numOfEdges--;}elseif(curr!=NULL&&curr->vertexNum==v2&&pre!=NULL){/*v1的邻接点v2在其它位置*/pre->next=curr->next;free(curr);G->numOfEdges--;}elseprintf("边<v1,v2>不存在!");}60第8章图6.寻找第一个邻接顶点寻找第一个邻接顶点可表示为:intGetFirstVex(AdjLGraphG,intv);其中参数G表示指定的邻接表,其类型为AdjGraph,参数v表示顶点编号,其类型为int。函数的功能:在图G中寻找序号为v的顶点的第一个邻接顶点,如果这样的邻接顶点存在则返回该邻接顶点的编号,否则返回-1。处理过程为:若v合法,在邻接表的第v行p=G.a[v].adj,若p!=NULL,返回p->vertexNum,否则返回-1。61第8章图intGetFirstVex(AdjLGraphG,intv)/*取邻接表中第一个邻接顶点编号*/{Edge*p;if(v<0||v>=G.numOfVerts){printf("参数v1或v2越界出错!");exit(0);}p=G.a[v].adj;if(p!=NULL)returnp->vertexNum;elsereturn-1;}62第8章图6.取邻接表中v2之后邻接顶点编号取邻接表中v2之后邻接顶点编号可表示为:intGetNextVex(AdjLGraphG,intv1,intv2);其中参数G表示指定的邻接表,其类型为AdjGraph,参数v1,V2表示顶点编号,其类型为int。函数的功能:在图G中寻找序号为v1的顶点的V2之后的邻接顶点,如果这样的邻接顶点存在则返回该邻接顶点的编号,否则返回-1。处理过程为:若v1,v2合法,在邻接表的第v行p=G.a[v].adj,若p->next!=NULL,p->vertexNum==v2,返回p->next->vertexNum,否则返回-1。63第8章图intGetNextVex(AdjLGraphG,intv1,constintv2)/*取邻接表中v2之后邻接顶点编号*/{Edge*p;if(v1<0||v1>G.numOfVerts||v2<0||v2>G.numOfVerts){printf("参数v1或v2越界出错!");exit(0);}p=G.a[v1].adj;64第8章图while(p!=NULL){if(p->vertexNum!=v2){p=p->next;continue;}elsebreak;}p=p->next;if(p!=NULL)returnp->vertexNum;elsereturn-1;}65第8章图/*建立邻接链表头文件CreatGraph.h*/typedefstruct{introw;intcol;}RowCol;7.建立邻接表建立邻接表可表示为:voidCreatGraph(AdjLGraph*G,DataTypev[],intn,RowCold[],inte);其中参数G表示指定的邻接表,其类型为AdjGraph,参数V表示顶点数组,其类型为DataType,参数n表示顶点数,其类型为int,参数d表示边数组,其类型为RowColWeight,函数外定义,参数e表示边数,其类型为int。66第8章图函数的功能:在图G中插入n个顶点信息V和e条边信息。处理过程为:⑴邻接表初始化⑵插入顶点集⑶插入边集函数实现如下:voidCreatGraph(AdjLGraph*G,DataTypev[],intn,RowCold[],inte){/*建立邻接链表*/inti,k;AdjInitiate(G);/*初始化*/for(i=0;i<n;i++)InsertVertex(G,i,v[i]);/*插入顶点集*/for(k=0;k<e;k++)InsertEdge(G,d[k].row,d[k].col);/*插入边集*/}67第8章图有兴趣的读者,可以用下面函数测试,看一看执行的结果。/*测试函数*/#include<stdio.h>#include<stdlib.h>typedefcharDataType;#defineMaxVertices100#include"AdjLGraph.h"#include"CreatGraph.h"voidmain(void){AdjLGraphg;chara[]={'A','B','C','D','E'};RowColrc[]={{0,1},{1,3},{3,2},{2,1},{0,4}};inti,n=5,e=5;Edge*p;CreatGraph(&g,a,n,rc,e);printf("%d%d\n",g.numOfVerts,g.numOfEdges);68第8章图for(i=0;i<g.numOfVerts;i++)/*输出每一个顶点的邻接顶点编号*/{printf("%c",g.a[i].data);p=g.a[i].adj;while(p!=NULL){printf("%d",p->vertexNum);p=p->next;}printf("\n");}DeleteEdge(&g,1,3);/*删除一条边*/printf("%d%d\n",g.numOfVerts,g.numOfEdges);69第8章图for(i=0;i<g.numOfVerts;i++){printf("%c",g.a[i].data);p=g.a[i].adj;while(p!=NULL){printf("%d",p->vertexNum);p=p->next;}printf("\n");}AdjDestroy(&g);}70第8章图8.2.3图的ADT定义以上我们讨论了图的结构特征、存储方式及其相关操作的实现,在本节中我们将给出线性表的ADT定义即抽象数据类型定义,为以后面向对象程序设计中的类定义奠定基础。根据面向对象程序设计的原则,实现部分与接口部分两者应该分离。接口部分可以用ADT定义即抽象数据类型定义来进行描述。一种数据类型的ADT定义由数据元素、结构及操作三部分组成。元素:ai同属于一个数据元素类型,i=1,2,…,n(n≥0);结构:对所有的数据元素ai,aj(i,j=1,2,…,n-1、i≠j)存在关系<ai,aj>或(ai,aj);71第8章图对图可执行基本操作为:⑴Initiate(G):初始化,建立一个空的图G。⑵Destroy(G):销毁,释放图G空间。⑶InsertVertex(G,i,vertex):插入顶点,图G插入一个顶点vertex,编号为i。⑷InsertEdge(G,v1,v2):插入边,构成边的两个顶点v1和v2是图中的顶点,则在图G中插入一条边。⑸DeleteEdge(G,v1,v2):删除边,若构成边的两个顶点v1和v2是图中的顶点,则在图G中删去边。⑹GetFirstVex(G,v):取第一个邻接顶点,给出顶点位置为v的第一个邻接顶点的位置,如果找不到,则函数返回-1。⑺GetNextVex(G,v1,v2):取下一个邻接顶点,给出顶点位置为v1的某邻接顶点v2的下—个邻接顶点的位置,如果找不到,则返回-1。通过前面的学习,我们知道,同样的操作会因为存储结构的不同而不同。72第8章图8.3遍历图给定一个无向连通图G,G=(V,E),当从V(G)中的任一顶点v出发,去访问图中其余顶点,使每个顶点仅被访问一次。这个过程叫做图的遍历。73图8.12存在回路的图第8章图然而,图的遍历要比树的遍历复杂的多。因为图中的任一顶点都可能和其余的顶点相邻接。所以在访问某个顶点后,可能沿着某条路径搜索之后,又回到该顶点。例如图8.12,由于图中存在回路,因此在访问了1,2,4,3之后沿着边(3,1)又可访问到1。为了避免同一顶点被访问多次,在遍历图的过程中,必须记下每个已访问过的顶点。为此,我们设一个辅助数组,可以利用表头向量a,让其tag域作为标志域
1已访问过即AdjLHeighta[v].tag=0未访问过通常有两种遍历图的方法,一种是深度优先搜索法,另一种为广度优先搜索法。74第8章图8.3.1深度优先搜索法设有无向连通图G(V,E),从V(G)中任一顶点V出发深度优先搜索法进行遍历的步骤是:先访问指定顶点v,然后访问与该顶点v相邻的顶点vt,再从vt出发,访问与Vt相邻且未被访问过的任意顶点,然后从此顶点出发,重复上述过程,直到一个所有邻接点都被访问过为止,然后回溯到此顶点的直接前趋,从这儿出发再继续访问。显然搜索是一个递归过程。对于图8.9(a),邻接表为图8.9(b),先选取A顶点作为搜索的起始点。顶点A的相邻顶点分别是C、B,所以沿着它的一个相邻顶点往下走。当走到顶C时,它有一个相邻顶点A,而A已被访问过所以应原路返回A,访问A的下一个相邻顶点B,往下走到D,而D有一个相邻顶点B已被访问过所以应原路返回B,其相邻顶点已被访问,原路返回上层A,上层为空则结束。75第8章图深度优先遍历递归函数可表示为:DepthFirstSearch(AdjLGraphgintv);其中参数g表示指定的邻接表,其类型为AdjGraph,v为起始顶点编号。函数的功能:从v开始对邻接表g所表示的图递归地进行深度优先遍历。处理过程为:⑴访问顶点v,并作访问标记;⑵按邻接取v的下一个未访问的邻接点w,并从w出发进行深度优先遍历;⑶重复⑵直至v的所有邻接点均被访问。76第8章图程序如下:DepthFirstSearch(AdjLGraphgintv)/*从某顶点v出发深度优先搜索子函数*/{intw;printf(“%d”,v);/*输出顶点*/g.a[v].tag=1;/*顶点标志域置1*/while(ptr[v]!=NULL){w=ptr[v]->vextexNum;/*取出顶点V的某相邻顶点的序号*/if(g.a[w].tag==0)DepthFirstSearch(g,w);/*如果该顶点未被访问过则递归调用,从该顶点w出发,*//*沿着它的各相邻顶点向下搜索*/77第8章图ptr[v]=ptr[v]->next;/*若从顶点v出发沿着某个相邻顶点的向下搜索已走到头,*//*则换一个相邻顶点,沿着它往下搜索*/}/*从顶点v出发对各相邻顶点逐个搜索,*//*直至从顶点v出发的所有并行路线已被搜索*/return;}78第8章图有兴趣的读者,可以用下面函数测试,看一看执行的结果。include<stdio.h>#include<stdlib.h>typedefcharDataType;#defineMaxVertices100#include"AdjLGraph.h"#include"CreatGraph.h"Edge*ptr[MaxVertices]voidmain(void){AdjLGraphg;chara[]={'A','B','C','D','E'};RowColrc[]={{0,1},{1,3},{3,2},{2,1},{0,4}};inti,w,v,n=5,e=5;Edge*p;79第8章图CreatGraph(&g,a,n,rc,e);for(v=0;v<=n;v++)*/初始化指针数组及标志域/*{ptr[v]=g.a[v].adj;g.a[v].tag=0;}for(v=0;v<n;v++)if(g.a[v].tag==0)DepthFirstSearch(g,v);}请读者注意:①在主函数中一定要初始化指针数组及标志域,否则程序无法执行。②主函数中要从每一个顶点出发作深度优先遍历,因为图有可能不连通。80第8章图8.3.2广度优先搜索设无向图G(V,E)是连通的,从V(G)中的任一顶点v出发按广度优先搜索法遍历图的步骤是:首先访问指定的起始顶点v,从v出发,依次访问与v相邻的未被访问过的顶点w1,w2,w3,...,wt然后依次从w1,w2,w3,...,wt出发,重复上述访问过程,直到所有顶点都被访问过为止。以图8.13为例,如选顶点1为起始点先进行访问。顶点有三个相邻顶点,依次是4、3、2,则广度优先搜索时对这几个顶点依次访问,然后再将这些相邻顶点的第一相邻顶点4作为起始顶点重复上述步骤,再优先访问6。再将第二个相邻顶点3作为起始顶点,重复上述步骤,顶点3有三个相邻顶点依次为6、5、2,其中2、6被访问过(不再访问),所以访问5。依次再判2、6、5,再重复上述步骤,因相邻顶点都被访问过即结束。81第8章图82广度优先遍历递归函数可表示为:voidBroadFirstSearch(AdjLGraphg,intv);其中参数g表示指定的邻接表,其类型为AdjGraph,v为起始顶点编号。函数的功能:从v开始对邻接表g所表示的图递归地进行广度优先遍历。图8.13广度优先搜索示意图第8章图处理过程为:⑴初始化队列;⑵访问顶点v,并作访问标记,v进入队列;⑶从队列中取出一个顶点,依次访问、标记它的每一个未访问的邻接点,并进入队列;重复⑵⑶直至队列为空。程序如下:voidBroadFirstSearch(AdjLGraphg,intv)/*从v出发广度优先搜索函数*/{Edge*ptr;SeqCQueueq;intv1,w;QueueInitiate(&q);printf(“%c”v);/*输出该顶点*/g.a[v]•tag=1;/*标志域置1*/QueueAppend(&q,v);/*将该顶点入队尾*/83第8章图while(QueueDelete(&q,&v1)!=0)/*while循环使属于同一层顶点的相邻顶点的依次出队。由于这些相邻顶点*//*作为上一层顶点均被访问过,出队的目的是访问它的下层相邻顶点*/{ptr=g.a[v1].adj;/*取出该顶点的第一个相邻顶点地址*/while(ptr!=NULL)/*while循环依次访问各相邻顶点*/{w=ptr->vertexNum;/*取出该顶点的序号*/ptr=ptr->next;/*从邻接表的相应链表中,取出下一个相邻顶点的地址以备访问*/if(g.a[w].tag==0)/*若该相邻顶点未被访问过*/{printf(“%d”w)/*则访问该相邻顶点*/g.a[w].tag=1;/*修改顶点标志域为1*/QueueAppend(&q,w);/*将访问过的顶点入队,以备在进入下一层搜索时使用*/}}}}84第8章图有兴趣的读者,可以用下面函数测试,看一看执行的结果。#include<stdlib.h>#include<malloc.h>typedefcharDataType;#defineMaxVertices100#include"AdjLGraph.h"#include"CreatGraph.h"#include"SeqCQueue.h"voidmain(void){AdjLGraphg;chara[]={'A','B','C','D','E'};RowColrc[]={{0,1},{1,3},{3,2},{2,1},{0,4}};inti,w,v,n=5,e=5;Edge*p;85第8章图CreatGraph(&g,a,n,rc,e);for(v=0;v<n;v++)*/初始化标志域/*g.a[v].tag=0;for(v=1;v<=n;v++)if(g.a[v].tag==0)BroadFirstSearch(g,v);}请读者注意:①在主函数中一定要初始化指针数组及标志域,否则程序无法执行。②主函数中要从每一个顶点出发作深度优先遍历,因为图有可能不连通。86第8章图8.4最小生成树8.4.1最小生成树的基本概念一个有n个顶点的连通图的生成树是原图的极小连通子图,它包含原图中的所有n个顶点,并且有保持图连通的最少的边。由生成树的定义可知:⑴若在生成树中删除一条边,就会使该生成树因变成非连通图而不再满足生成树的定义;⑵若在生成树中增加一条边,就会使该生成树中因存在回路而不再满足生成树的定义;⑶一个连通图的生成树可能有许多。使用不同的遍历图的方法可以得到不同的生成树。如对无向图使用深度优先搜索遍历方法与使用广度优先搜索遍历方法将会得到两个结构不同的生成树;从不同的顶点出发,也可以得到不同的生成树。图8.14给出了一个无向图和它的两棵不同的生成树。87第8章图88图8.14无向图和它的不同的生成树(a)无向图;(b)生成树1;(c)生成树2第8章图从生成树的定义显然可以证明,对于有n个顶点的无向连通图,无论它的生成树的形状如何,一定有且仅有n-1条边。如果无向连通图是一个网(或称带权图),那么它的所有生成树中必有一棵边的权值总和最小的生成树,我们称这棵生成树为最小代价生成树,简称最小生成树。对于一个带权的连通图(即网络),如何找出一棵生成树,使得各边上的权值总和达到最小,这是一个有实际意义的问题。例如,在n个城市之间建立通信网络,至少要架设n-1条线路。这时自然会考虑:如何做才能使得总造价最少?89第8章图在每两个城市之间都可以架设一条通信线路,并要花费一定的代价。若用图的顶点表示n个城市,用边表示两城市之间架设的线路,用边上的权值表示架设该线路的造价,就可以建立一个通信网络。对于这样一个有n个顶点的网络,可以有不同的生成树,每一棵生成树都可以构成通信网络。希望能够根据各条边上的权值,选择一棵总造价最小的生成树,这就是构造网络的最小(代价)生成树(MinimumCostSpanningTree)的问题。从最小生成树的定义可知,构造有n个顶点的无向连通网的最小生成树必须满足以下三条:(1)构造的最小生成树必须包括n个顶点;(2)构造的最小生成树中有且只有n-1条边;(3)构造的最小生成树中不存在回路。构造的最小生成树的方法有许多种,典型的构造方法有两种:一种称作普里姆(Prim)算法,另一种称作克鲁斯卡尔(Kruskal)算法。90第8章图8.4.2普里姆算法假设G=(V,E)是一个网络,其中V为网中顶点的集合,E为网中带权边的集合。设置两个新的集合U和T,其中U用于存放网G的最小生成树的顶点的集合,T用于存放网G的最小生成树的带权边的集合。令集合U的初值为U={u0}(即假设构造最小生成树时均从顶点u0开始),集合T的初值为T={}。普里姆算法的思想是:从所有顶点u∈U和顶点v∈V-U的带权边中选出具有最小权值的边(u,v)∈U,将顶点v加入集合U中,将边(u,v)加入集合T中。如此不断重复,直到U=V时即构造完毕。此时集合U中存放着最小生成树的顶点的集合,集合T中存放着最小生成树的带权边的集合。91第8章图图8.15给出了用普里姆算法构造最小生成树的过程。图8.15(a)是一个有7个顶点、10条无向边的带权图。初始时算法的集合U={A},集合V-U={B,C,D,E,F,G},T={},如图8.15(b)所示。在所有u为集合U中顶点,v为集合V-U中顶点的边(u,v)中,寻找具有最小权值的边(u,v),寻找到的是边(A,B),权为50,把顶点B从集合V—U加入到集合U中,把边(A,B)加入到了中,如图8.15(c)所示。在所有u为集合U中顶点,v为集合V-U中顶点的边(u,v),中寻找具有最小权值的边(u,v),寻找到的是边(B,E),权为40,把顶点E从集合V-U加入到集合U中,把边(B,E)加入到T中,如图8.15(d)所示;随后依次从集合V-U加入到集合U中的顶点为D,F,G,C,依次加入到T中的边为(E,D),权为50,(D,F)权为30,(D,G)权为42,(G,C)权为45,分别如图8.15(e),(f),(g),(h)所示。最后得到的图8.15(h)就是原带权连通图的最小生成树。92第8章图93图8.15普里姆算法构造最小生成树的过程第8章图
我们再来讨论普里姆算法的实现问题。普里姆算法要处理的带权图G可以是前面讨论过的任意存储结构的图。我们下面讨论的普里姆算法使用邻接矩阵作为图的存储结构,算法中定义的图G的顶点集合就是普里姆算法中的集合V,图G的边集合就是普里姆算法中的集合E。普里姆算法所构造的最小生成树是集合U和集合T,集合U是顶点的集合,集合T是带权边的集合。94第8章图
为实现普里姆算法,我们还需定义一个临时数组lowCost[n],用来保存集合U中顶点u与集合V-U中顶点v的所有边中当前具有最小权值的边(u,v)。回顾带权图中权的定义,当弧头顶点等于弧尾顶点时权值等于0(即邻接矩阵的对角线元素值为0),我们在此可以利用权定义中的这一点,令lowCost的初始值为邻接矩阵中第0行的值,lowCost表示了从集合U中第一个顶点到达集合V-U中各个顶点的代价。然后我们从lowCost中寻找具有最小权值的边,这样的边也就意味着其弧头顶点为集合U中顶点,其弧尾顶点为集合V-U中顶点。每选到一条这样的边(u,v),就将lowCost[v]置为0,表示顶点v已从集合V-U加入到集合U中。如果顶点v从集合V-U进入集合U后,当前lowCost中从集合U中顶点到达集合V-U中顶点存在更小代价的边时,则用这样的边的权值修改原来的权值。最后输出即为所构造的最小生成树。95第8章图用普里姆方法建立网G的最小生成树minSTree的函数如下:voidPrim(AdjMWGraphG)/*用普里姆方法建立网G的最小生成树closeVertex*/{intn=G.Vertices.size,minCost;int*lowCost=(int*)malloc(sizeof(int)*n);inti,j,k;/*从序号为0的顶点出发构造最小生成树*/printf("顶点值=%c\n",G.Vertices.list[0]);for(i=1;i<n;i++) lowCost[i]=G.edge[0][i];lowCost[0]=0;for(i=1;i<n;i++){96第8章图/*寻找当前最小权值的边的顶点*/minCost=MaxWeight;/*MaxWeight为定义的最大权值*/j=1;k=1;while(j<n){if(lowCost[j]<minCost&&lowCost[j]!=0){minCost=lowCost[j];k=j;}j++;}printf("顶点值=%c边的权值=%d\n",G.Vertices.list[k],minCost);lowCost[k]=-1;97
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 【正版授权】 ISO/TS 21219-13:2025 EN Intelligent transport systems - Traffic and travel information via transport protocol experts group,generation 2 (TPEG2) - Part 13: Public transport
- 2025标准农田建设合同管理的任务和方法
- 2025员工个人劳动合同
- 2025年度房屋置换增值服务全新房屋置换对方房屋置换及增值合同3篇
- 2025农村合作建房项目施工临时用电安全合同2篇
- 二零二五年度公司股权转让及后续管理服务合同3篇
- 二零二五年度房地产项目合作合同解除协议2篇
- 二零二五年度农产品批发市场租赁合作合同3篇
- 二零二五年度智能家居产品开发合作协议书3篇
- 二零二五年度办公室租赁合同模板:含员工激励及福利计划3篇
- 部编人教版四年级数学上册期末考试卷(可打印)
- 一例阿尔茨海默病患者的护理查房
- 农贸市场安全生产工作方案
- 咸阳租房合同
- 《钢筋保护层检测》课件
- YJ-T 27-2024 应急指挥通信保障能力建设规范
- 合伙人协议书决策机制
- 西藏畜牧兽医知识培训课件
- 护理专业人才培养方案论证报告
- 我的家乡武汉
- 眼镜制造业灌胶机市场前景与机遇分析
评论
0/150
提交评论