• 图论算法讲解


    图的基本概念

    图是由顶点的有穷非空集合和顶点之间边的集合组成,通常表示为:G=(V,E),其中 V 是非空有限集合, 代表顶点,E 是可以为空的有限集合,代表边。

    若顶点 vi 和 vj 间的边没有方向,则称这条边为无向边,用无序偶对 (vi,vj) 表示;若顶点 vi 和 vj 间的边有方向,则称这条边为有向边(弧),用有序偶对 表示,其中 vi 称为弧头,vj 称为弧尾。

    图的存储结构

    图的存储结构分为邻接矩阵、邻接表、链式向前星、十字链表、邻接多重表。

    邻接矩阵

    图的邻接矩阵存储也称数组表示法,其方法是用一个一维数组存储图中的顶点,用一个二维数组存储图中的所有的边,存储顶点之间邻接关系的二维数组称为邻接矩阵

    设图 G=(V,E) 具有 n 个顶点,则其邻接矩阵是一个 n*n 的方阵,定义为

    G[i][j]=\left\{\begin{matrix}1,(v_i,v_j)\in E\:\:or\:\:<v_i,v_j>\in E \\ 0, (v_i,v_j)\notin E\:\:and\:\:<v_i,v_j>\notin E \end{matrix}\right.

     其对确定边操作效率高,但由于其过高的空间复杂度,对稀疏图来说造成了极大的浪费,且由于二维数组可开的空间范围有限,因此邻接矩阵一般适用于点较少的图。

    1. #include "stdio.h"
    2. #include "stdlib.h"
    3. #include "io.h"
    4. #include "math.h"
    5. #include "time.h"
    6. #define OK 1
    7. #define ERROR 0
    8. #define TRUE 1
    9. #define FALSE 0
    10. #define MAXVEX 100 /* 最大顶点数,应由用户定义 */
    11. #define INFINITY 65535
    12. typedef int Status; /* Status是函数的类型,其值是函数结果状态代码,如OK等 */
    13. typedef char VertexType; /* 顶点类型应由用户定义 */
    14. typedef int EdgeType; /* 边上的权值类型应由用户定义 */
    15. typedef struct
    16. {
    17. VertexType vexs[MAXVEX]; /* 顶点表 */
    18. EdgeType arc[MAXVEX][MAXVEX];/* 邻接矩阵,可看作边表 */
    19. int numNodes, numEdges; /* 图中当前的顶点数和边数 */
    20. }MGraph;
    21. /* 建立无向网图的邻接矩阵表示 */
    22. void CreateMGraph(MGraph *G)
    23. {
    24. int i,j,k,w;
    25. printf("输入顶点数和边数:\n");
    26. scanf("%d,%d",&G->numNodes,&G->numEdges); /* 输入顶点数和边数 */
    27. for(i = 0;i numNodes;i++) /* 读入顶点信息,建立顶点表 */
    28. scanf(&G->vexs[i]);
    29. for(i = 0;i numNodes;i++)
    30. for(j = 0;j numNodes;j++)
    31. G->arc[i][j]=INFINITY; /* 邻接矩阵初始化 */
    32. for(k = 0;k numEdges;k++) /* 读入numEdges条边,建立邻接矩阵 */
    33. {
    34. printf("输入边(vi,vj)上的下标i,下标j和权w:\n");
    35. scanf("%d,%d,%d",&i,&j,&w); /* 输入边(vi,vj)上的权w */
    36. G->arc[i][j]=w;
    37. G->arc[j][i]= G->arc[i][j]; /* 因为是无向图,矩阵对称 */
    38. }
    39. }
    40. int main(void)
    41. {
    42. MGraph G;
    43. CreateMGraph(&G);
    44. getchar();
    45. return 0;
    46. }

    邻接表

    邻接表是一种顺序存储与链接存储相结合的存储方法,类似于树的孩子链表表示法。

    对于图 G 的每个顶点 vi,将所有邻接于 vi 的顶点链成一个单链表,称为 vi 的边表,若图 G 是一有向图,则称为 vi 的出边表。

    为方便对所有边表的头指针进行存储操作,使用顺序存储,这样,存储边表头指针的数组和存储顶点信息的数组构成了邻接表的表头数组,称为顶点表。 

     相较于邻接矩阵,其由数组转链表,由定长转不定长,内存利用率高,能较好的处理稀疏图,对不确定的边操作较为方便,但由于存储方式的问题,其对重边不好处理,且对确定边处理效率并不高。

    1. #include "stdio.h"
    2. #include "stdlib.h"
    3. #include "io.h"
    4. #include "math.h"
    5. #include "time.h"
    6. #define OK 1
    7. #define ERROR 0
    8. #define TRUE 1
    9. #define FALSE 0
    10. #define MAXVEX 100 /* 最大顶点数,应由用户定义 */
    11. typedef int Status; /* Status是函数的类型,其值是函数结果状态代码,如OK等 */
    12. typedef char VertexType; /* 顶点类型应由用户定义 */
    13. typedef int EdgeType; /* 边上的权值类型应由用户定义 */
    14. typedef struct EdgeNode /* 边表结点 */
    15. {
    16. int adjvex; /* 邻接点域,存储该顶点对应的下标 */
    17. EdgeType info; /* 用于存储权值,对于非网图可以不需要 */
    18. struct EdgeNode *next; /* 链域,指向下一个邻接点 */
    19. }EdgeNode;
    20. typedef struct VertexNode /* 顶点表结点 */
    21. {
    22. VertexType data; /* 顶点域,存储顶点信息 */
    23. EdgeNode *firstedge;/* 边表头指针 */
    24. }VertexNode, AdjList[MAXVEX];
    25. typedef struct
    26. {
    27. AdjList adjList;
    28. int numNodes,numEdges; /* 图中当前顶点数和边数 */
    29. }GraphAdjList;
    30. /* 建立图的邻接表结构 */
    31. void CreateALGraph(GraphAdjList *G)
    32. {
    33. int i,j,k;
    34. EdgeNode *e;
    35. printf("输入顶点数和边数:\n");
    36. scanf("%d,%d",&G->numNodes,&G->numEdges); /* 输入顶点数和边数 */
    37. for(i = 0;i < G->numNodes;i++) /* 读入顶点信息,建立顶点表 */
    38. {
    39. scanf(&G->adjList[i].data); /* 输入顶点信息 */
    40. G->adjList[i].firstedge=NULL; /* 将边表置为空表 */
    41. }
    42. for(k = 0;k < G->numEdges;k++)/* 建立边表 */
    43. {
    44. printf("输入边(vi,vj)上的顶点序号:\n");
    45. scanf("%d,%d",&i,&j); /* 输入边(vi,vj)上的顶点序号 */
    46. e=(EdgeNode *)malloc(sizeof(EdgeNode)); /* 向内存申请空间,生成边表结点 */
    47. e->adjvex=j; /* 邻接序号为j */
    48. e->next=G->adjList[i].firstedge; /* 将e的指针指向当前顶点上指向的结点 */
    49. G->adjList[i].firstedge=e; /* 将当前顶点的指针指向e */
    50. e=(EdgeNode *)malloc(sizeof(EdgeNode)); /* 向内存申请空间,生成边表结点 */
    51. e->adjvex=i; /* 邻接序号为i */
    52. e->next=G->adjList[j].firstedge; /* 将e的指针指向当前顶点上指向的结点 */
    53. G->adjList[j].firstedge=e; /* 将当前顶点的指针指向e */
    54. }
    55. }
    56. int main(void)
    57. {
    58. GraphAdjList G;
    59. CreateALGraph(&G);
    60. return 0;
    61. }

    这里对边表结点的赋值使用到了头插法

    十字链表与邻接多重表也是图的一种存储方法,但由于操作复杂、内存利用率不高等因素常不被使用。

    十字链表

    十字链表是有向图的一种存储方法,其实质上是邻接表与逆邻接表的结合,每条边对应的边结点分别组织到出边表与入边表中,其顶点表和边表结点结构如下:

    对于顶点表来说,data 为数据域,存放顶点信息;firstIn 为入边表头指针,指向以该顶点为终点的弧构成的链表中的第一个结点;firstOut 为出边表头指针,指向以该顶点为始点的弧构成的链表中的第一个结点。

    对于边表来说,tailPos 代表弧的起点在顶点表中的下标;headPos 代表弧的终点在顶点表中的下标;headLink 为入边表指针域,指向终点相同的下一条边;tailLink 为出边表指针域,指向起点相同的下一条边。

    用邻接表存储无向图,每条边的两个顶点分别在以该边所依附的两个顶点的边表中,这种重复存储给图的某些操作带来不便,因此有了邻接多重表。

    邻接多重表

    邻接多重表主要用于存储无向图,其存储结构与邻接表类似,也有顶点表和边表组成,每条边用一个边表结点表示,其顶点表和边表的结点结构如下:

     

    对于顶点表而言,data  为数据域,存放顶点信息;firstEdge 为边表头指针,指向依附于该顶点的第一条边的边表结点。

    对于边表而言,iPos、jPos 为与某边依附的两顶点在顶点表中的下标;iLink 为指针域,指向依附于顶点 iPos 的下一条边;jLink 为指针域,指向依附于顶点 jPos 的下一条边

    图的搜索

    根据搜索方法的不同,分为深度优先遍历(DFS)、广度优先遍历(BFS),两者时间复杂度都是 O(N*N),通常采用深度优先遍历。

    深度优先遍历

    图的深度优先遍历的基本过程为:

    1.从图中某个顶点 v0 出发,首先访问 v0
    2.访问结点 v0 的第一个邻接点,以这个邻接点 vt 作为一个新节点,访问 vt 所有邻接点,直到以 vt 出发的所有节点都被访问
    3.回溯到 v0 的下一个未被访问过的邻接点,以这个邻结点为新节点,重复步骤 2,直到图中所有与 v0 相通的所有节点都被访问
    4.若此时图中仍有未被访问的结点,则另选图中的一个未被访问的顶点作为起始点,重复步骤 1,直到图中的所有节点均被访问

     深度优先遍历其实就是递归过程。这里就像是树的前序遍历。

    1. #include "stdio.h"
    2. #include "stdlib.h"
    3. #include "io.h"
    4. #include "math.h"
    5. #include "time.h"
    6. #define OK 1
    7. #define ERROR 0
    8. #define TRUE 1
    9. #define FALSE 0
    10. typedef int Status; /* Status是函数的类型,其值是函数结果状态代码,如OK等 */
    11. typedef int Boolean; /* Boolean是布尔类型,其值是TRUE或FALSE */
    12. typedef char VertexType; /* 顶点类型应由用户定义 */
    13. typedef int EdgeType; /* 边上的权值类型应由用户定义 */
    14. #define MAXSIZE 9 /* 存储空间初始分配量 */
    15. #define MAXEDGE 15
    16. #define MAXVEX 9
    17. #define INFINITY 65535
    18. typedef struct
    19. {
    20. VertexType vexs[MAXVEX]; /* 顶点表 */
    21. EdgeType arc[MAXVEX][MAXVEX];/* 邻接矩阵,可看作边表 */
    22. int numVertexes, numEdges; /* 图中当前的顶点数和边数 */
    23. }MGraph;
    24. /* 用到的队列结构与函数********************************** */
    25. /* 循环队列的顺序存储结构 */
    26. typedef struct
    27. {
    28. int data[MAXSIZE];
    29. int front; /* 头指针 */
    30. int rear; /* 尾指针,若队列不空,指向队列尾元素的下一个位置 */
    31. }Queue;
    32. /* 初始化一个空队列Q */
    33. Status InitQueue(Queue *Q)
    34. {
    35. Q->front=0;
    36. Q->rear=0;
    37. return OK;
    38. }
    39. /* 若队列Q为空队列,则返回TRUE,否则返回FALSE */
    40. Status QueueEmpty(Queue Q)
    41. {
    42. if(Q.front==Q.rear) /* 队列空的标志 */
    43. return TRUE;
    44. else
    45. return FALSE;
    46. }
    47. /* 若队列未满,则插入元素e为Q新的队尾元素 */
    48. Status EnQueue(Queue *Q,int e)
    49. {
    50. if ((Q->rear+1)%MAXSIZE == Q->front) /* 队列满的判断 */
    51. return ERROR;
    52. Q->data[Q->rear]=e; /* 将元素e赋值给队尾 */
    53. Q->rear=(Q->rear+1)%MAXSIZE;/* rear指针向后移一位置, */
    54. /* 若到最后则转到数组头部 */
    55. return OK;
    56. }
    57. /* 若队列不空,则删除Q中队头元素,用e返回其值 */
    58. Status DeQueue(Queue *Q,int *e)
    59. {
    60. if (Q->front == Q->rear) /* 队列空的判断 */
    61. return ERROR;
    62. *e=Q->data[Q->front]; /* 将队头元素赋值给e */
    63. Q->front=(Q->front+1)%MAXSIZE; /* front指针向后移一位置, */
    64. /* 若到最后则转到数组头部 */
    65. return OK;
    66. }
    67. /* ****************************************************** */
    68. void CreateMGraph(MGraph *G)
    69. {
    70. int i, j;
    71. G->numEdges=15;
    72. G->numVertexes=9;
    73. /* 读入顶点信息,建立顶点表 */
    74. G->vexs[0]='A';
    75. G->vexs[1]='B';
    76. G->vexs[2]='C';
    77. G->vexs[3]='D';
    78. G->vexs[4]='E';
    79. G->vexs[5]='F';
    80. G->vexs[6]='G';
    81. G->vexs[7]='H';
    82. G->vexs[8]='I';
    83. for (i = 0; i < G->numVertexes; i++)/* 初始化图 */
    84. {
    85. for ( j = 0; j < G->numVertexes; j++)
    86. {
    87. G->arc[i][j]=0;
    88. }
    89. }
    90. G->arc[0][1]=1;
    91. G->arc[0][5]=1;
    92. G->arc[1][2]=1;
    93. G->arc[1][8]=1;
    94. G->arc[1][6]=1;
    95. G->arc[2][3]=1;
    96. G->arc[2][8]=1;
    97. G->arc[3][4]=1;
    98. G->arc[3][7]=1;
    99. G->arc[3][6]=1;
    100. G->arc[3][8]=1;
    101. G->arc[4][5]=1;
    102. G->arc[4][7]=1;
    103. G->arc[5][6]=1;
    104. G->arc[6][7]=1;
    105. for(i = 0; i < G->numVertexes; i++)
    106. {
    107. for(j = i; j < G->numVertexes; j++)
    108. {
    109. G->arc[j][i] =G->arc[i][j];
    110. }
    111. }
    112. }
    113. Boolean visited[MAXVEX]; /* 访问标志的数组 */
    114. /* 邻接矩阵的深度优先递归算法 */
    115. void DFS(MGraph G, int i)
    116. {
    117. int j;
    118. visited[i] = TRUE;
    119. printf("%c ", G.vexs[i]);/* 打印顶点,也可以其它操作 */
    120. for(j = 0; j < G.numVertexes; j++)
    121. if(G.arc[i][j] == 1 && !visited[j])
    122. DFS(G, j);/* 对为访问的邻接顶点递归调用 */
    123. }
    124. /* 邻接矩阵的深度遍历操作 */
    125. void DFSTraverse(MGraph G)
    126. {
    127. int i;
    128. for(i = 0; i < G.numVertexes; i++)
    129. visited[i] = FALSE; /* 初始所有顶点状态都是未访问过状态 */
    130. for(i = 0; i < G.numVertexes; i++)
    131. if(!visited[i]) /* 对未访问过的顶点调用DFS,若是连通图,只会执行一次 */
    132. DFS(G, i);
    133. }
    134. /* 邻接矩阵的广度遍历算法 */
    135. void BFSTraverse(MGraph G)
    136. {
    137. int i, j;
    138. Queue Q;
    139. for(i = 0; i < G.numVertexes; i++)
    140. visited[i] = FALSE;
    141. InitQueue(&Q); /* 初始化一辅助用的队列 */
    142. for(i = 0; i < G.numVertexes; i++) /* 对每一个顶点做循环 */
    143. {
    144. if (!visited[i]) /* 若是未访问过就处理 */
    145. {
    146. visited[i]=TRUE; /* 设置当前顶点访问过 */
    147. printf("%c ", G.vexs[i]);/* 打印顶点,也可以其它操作 */
    148. EnQueue(&Q,i); /* 将此顶点入队列 */
    149. while(!QueueEmpty(Q)) /* 若当前队列不为空 */
    150. {
    151. DeQueue(&Q,&i); /* 将队对元素出队列,赋值给i */
    152. for(j=0;j
    153. {
    154. /* 判断其它顶点若与当前顶点存在边且未访问过 */
    155. if(G.arc[i][j] == 1 && !visited[j])
    156. {
    157. visited[j]=TRUE; /* 将找到的此顶点标记为已访问 */
    158. printf("%c ", G.vexs[j]); /* 打印顶点 */
    159. EnQueue(&Q,j); /* 将找到的此顶点入队列 */
    160. }
    161. }
    162. }
    163. }
    164. }
    165. }
    166. int main(void)
    167. {
    168. MGraph G;
    169. CreateMGraph(&G);
    170. printf("\n深度遍历:");
    171. DFSTraverse(G);
    172. printf("\n广度遍历:");
    173. BFSTraverse(G);
    174. return 0;
    175. }

    广度优先遍历

    图的广度优先遍历的基本过程为:

    1.从图中某个顶点 v0 出发,首先访问 v0,将 v0 加入队列
    2.将队首元素的未被访问过的邻接点加入队列,访问队首元素并将队首元素出队,直到队列为空
    3.若此时图中仍有未被访问的结点,则另选图中的一个未被访问的顶点作为起始点,重复步骤 1,直到图中的所有节点均被访问过。

    1. #include "stdio.h"
    2. #include "stdlib.h"
    3. #include "io.h"
    4. #include "math.h"
    5. #include "time.h"
    6. #define OK 1
    7. #define ERROR 0
    8. #define TRUE 1
    9. #define FALSE 0
    10. #define MAXSIZE 9 /* 存储空间初始分配量 */
    11. #define MAXEDGE 15
    12. #define MAXVEX 9
    13. #define INFINITY 65535
    14. typedef int Status; /* Status是函数的类型,其值是函数结果状态代码,如OK等 */
    15. typedef int Boolean; /* Boolean是布尔类型,其值是TRUE或FALSE */
    16. typedef char VertexType; /* 顶点类型应由用户定义 */
    17. typedef int EdgeType; /* 边上的权值类型应由用户定义 */
    18. /* 邻接矩阵结构 */
    19. typedef struct
    20. {
    21. VertexType vexs[MAXVEX]; /* 顶点表 */
    22. EdgeType arc[MAXVEX][MAXVEX];/* 邻接矩阵,可看作边表 */
    23. int numVertexes, numEdges; /* 图中当前的顶点数和边数 */
    24. }MGraph;
    25. /* 邻接表结构****************** */
    26. typedef struct EdgeNode /* 边表结点 */
    27. {
    28. int adjvex; /* 邻接点域,存储该顶点对应的下标 */
    29. int weight; /* 用于存储权值,对于非网图可以不需要 */
    30. struct EdgeNode *next; /* 链域,指向下一个邻接点 */
    31. }EdgeNode;
    32. typedef struct VertexNode /* 顶点表结点 */
    33. {
    34. int in; /* 顶点入度 */
    35. char data; /* 顶点域,存储顶点信息 */
    36. EdgeNode *firstedge;/* 边表头指针 */
    37. }VertexNode, AdjList[MAXVEX];
    38. typedef struct
    39. {
    40. AdjList adjList;
    41. int numVertexes,numEdges; /* 图中当前顶点数和边数 */
    42. }graphAdjList,*GraphAdjList;
    43. /* **************************** */
    44. /* 用到的队列结构与函数********************************** */
    45. /* 循环队列的顺序存储结构 */
    46. typedef struct
    47. {
    48. int data[MAXSIZE];
    49. int front; /* 头指针 */
    50. int rear; /* 尾指针,若队列不空,指向队列尾元素的下一个位置 */
    51. }Queue;
    52. /* 初始化一个空队列Q */
    53. Status InitQueue(Queue *Q)
    54. {
    55. Q->front=0;
    56. Q->rear=0;
    57. return OK;
    58. }
    59. /* 若队列Q为空队列,则返回TRUE,否则返回FALSE */
    60. Status QueueEmpty(Queue Q)
    61. {
    62. if(Q.front==Q.rear) /* 队列空的标志 */
    63. return TRUE;
    64. else
    65. return FALSE;
    66. }
    67. /* 若队列未满,则插入元素e为Q新的队尾元素 */
    68. Status EnQueue(Queue *Q,int e)
    69. {
    70. if ((Q->rear+1)%MAXSIZE == Q->front) /* 队列满的判断 */
    71. return ERROR;
    72. Q->data[Q->rear]=e; /* 将元素e赋值给队尾 */
    73. Q->rear=(Q->rear+1)%MAXSIZE;/* rear指针向后移一位置, */
    74. /* 若到最后则转到数组头部 */
    75. return OK;
    76. }
    77. /* 若队列不空,则删除Q中队头元素,用e返回其值 */
    78. Status DeQueue(Queue *Q,int *e)
    79. {
    80. if (Q->front == Q->rear) /* 队列空的判断 */
    81. return ERROR;
    82. *e=Q->data[Q->front]; /* 将队头元素赋值给e */
    83. Q->front=(Q->front+1)%MAXSIZE; /* front指针向后移一位置, */
    84. /* 若到最后则转到数组头部 */
    85. return OK;
    86. }
    87. /* ****************************************************** */
    88. void CreateMGraph(MGraph *G)
    89. {
    90. int i, j;
    91. G->numEdges=15;
    92. G->numVertexes=9;
    93. /* 读入顶点信息,建立顶点表 */
    94. G->vexs[0]='A';
    95. G->vexs[1]='B';
    96. G->vexs[2]='C';
    97. G->vexs[3]='D';
    98. G->vexs[4]='E';
    99. G->vexs[5]='F';
    100. G->vexs[6]='G';
    101. G->vexs[7]='H';
    102. G->vexs[8]='I';
    103. for (i = 0; i < G->numVertexes; i++)/* 初始化图 */
    104. {
    105. for ( j = 0; j < G->numVertexes; j++)
    106. {
    107. G->arc[i][j]=0;
    108. }
    109. }
    110. G->arc[0][1]=1;
    111. G->arc[0][5]=1;
    112. G->arc[1][2]=1;
    113. G->arc[1][8]=1;
    114. G->arc[1][6]=1;
    115. G->arc[2][3]=1;
    116. G->arc[2][8]=1;
    117. G->arc[3][4]=1;
    118. G->arc[3][7]=1;
    119. G->arc[3][6]=1;
    120. G->arc[3][8]=1;
    121. G->arc[4][5]=1;
    122. G->arc[4][7]=1;
    123. G->arc[5][6]=1;
    124. G->arc[6][7]=1;
    125. for(i = 0; i < G->numVertexes; i++)
    126. {
    127. for(j = i; j < G->numVertexes; j++)
    128. {
    129. G->arc[j][i] =G->arc[i][j];
    130. }
    131. }
    132. }
    133. /* 利用邻接矩阵构建邻接表 */
    134. void CreateALGraph(MGraph G,GraphAdjList *GL)
    135. {
    136. int i,j;
    137. EdgeNode *e;
    138. *GL = (GraphAdjList)malloc(sizeof(graphAdjList));
    139. (*GL)->numVertexes=G.numVertexes;
    140. (*GL)->numEdges=G.numEdges;
    141. for(i= 0;i /* 读入顶点信息,建立顶点表 */
    142. {
    143. (*GL)->adjList[i].in=0;
    144. (*GL)->adjList[i].data=G.vexs[i];
    145. (*GL)->adjList[i].firstedge=NULL; /* 将边表置为空表 */
    146. }
    147. for(i=0;i/* 建立边表 */
    148. {
    149. for(j=0;j
    150. {
    151. if (G.arc[i][j]==1)
    152. {
    153. e=(EdgeNode *)malloc(sizeof(EdgeNode));
    154. e->adjvex=j; /* 邻接序号为j */
    155. e->next=(*GL)->adjList[i].firstedge; /* 将当前顶点上的指向的结点指针赋值给e */
    156. (*GL)->adjList[i].firstedge=e; /* 将当前顶点的指针指向e */
    157. (*GL)->adjList[j].in++;
    158. }
    159. }
    160. }
    161. }
    162. Boolean visited[MAXSIZE]; /* 访问标志的数组 */
    163. /* 邻接表的深度优先递归算法 */
    164. void DFS(GraphAdjList GL, int i)
    165. {
    166. EdgeNode *p;
    167. visited[i] = TRUE;
    168. printf("%c ",GL->adjList[i].data);/* 打印顶点,也可以其它操作 */
    169. p = GL->adjList[i].firstedge;
    170. while(p)
    171. {
    172. if(!visited[p->adjvex])
    173. DFS(GL, p->adjvex);/* 对为访问的邻接顶点递归调用 */
    174. p = p->next;
    175. }
    176. }
    177. /* 邻接表的深度遍历操作 */
    178. void DFSTraverse(GraphAdjList GL)
    179. {
    180. int i;
    181. for(i = 0; i < GL->numVertexes; i++)
    182. visited[i] = FALSE; /* 初始所有顶点状态都是未访问过状态 */
    183. for(i = 0; i < GL->numVertexes; i++)
    184. if(!visited[i]) /* 对未访问过的顶点调用DFS,若是连通图,只会执行一次 */
    185. DFS(GL, i);
    186. }
    187. /* 邻接表的广度遍历算法 */
    188. void BFSTraverse(GraphAdjList GL)
    189. {
    190. int i;
    191. EdgeNode *p;
    192. Queue Q;
    193. for(i = 0; i < GL->numVertexes; i++)
    194. visited[i] = FALSE;
    195. InitQueue(&Q);
    196. for(i = 0; i < GL->numVertexes; i++)
    197. {
    198. if (!visited[i])
    199. {
    200. visited[i]=TRUE;
    201. printf("%c ",GL->adjList[i].data);/* 打印顶点,也可以其它操作 */
    202. EnQueue(&Q,i);
    203. while(!QueueEmpty(Q))
    204. {
    205. DeQueue(&Q,&i);
    206. p = GL->adjList[i].firstedge; /* 找到当前顶点的边表链表头指针 */
    207. while(p)
    208. {
    209. if(!visited[p->adjvex]) /* 若此顶点未被访问 */
    210. {
    211. visited[p->adjvex]=TRUE;
    212. printf("%c ",GL->adjList[p->adjvex].data);
    213. EnQueue(&Q,p->adjvex); /* 将此顶点入队列 */
    214. }
    215. p = p->next; /* 指针指向下一个邻接点 */
    216. }
    217. }
    218. }
    219. }
    220. }
    221. int main(void)
    222. {
    223. MGraph G;
    224. GraphAdjList GL;
    225. CreateMGraph(&G);
    226. CreateALGraph(G,&GL);
    227. printf("\n深度遍历:");
    228. DFSTraverse(GL);
    229. printf("\n广度遍历:");
    230. BFSTraverse(GL);
    231. return 0;
    232. }

    AOV 网与拓扑排序

    AOV 网

    一个有向无环图称为无环图(Directed Acyclic Graph),简称 DAG 图,因此一个 AOV 网必定是一个有向无环图,即不带有回路。

    拓扑排序

    1. 选择一个入度为 0 的顶点并输出。
    2. 从 AOV 网中删除此顶点及以此顶点为起点的所有关联边(即相邻边的入度减1)。
    3. 重复上述两步,直到不存在入度为 0 的顶点为止。
    4. 若输出的顶点数小于 AOV 网中的顶点数,则说明 AOV 网中回路,不是一个标准的 AOV 网。

    整个拓扑排序的时间复杂度是o(n+e).  n个定点e条弧的AOV网。

    1. #include "stdio.h"
    2. #include "stdlib.h"
    3. #include "io.h"
    4. #include "math.h"
    5. #include "time.h"
    6. #define OK 1
    7. #define ERROR 0
    8. #define TRUE 1
    9. #define FALSE 0
    10. #define MAXEDGE 20
    11. #define MAXVEX 14
    12. #define INFINITY 65535
    13. typedef int Status; /* Status是函数的类型,其值是函数结果状态代码,如OK等 */
    14. /* 邻接矩阵结构 */
    15. typedef struct
    16. {
    17. int vexs[MAXVEX];
    18. int arc[MAXVEX][MAXVEX];
    19. int numVertexes, numEdges;
    20. }MGraph;
    21. /* 邻接表结构****************** */
    22. typedef struct EdgeNode /* 边表结点 */
    23. {
    24. int adjvex; /* 邻接点域,存储该顶点对应的下标 */
    25. int weight; /* 用于存储权值,对于非网图可以不需要 */
    26. struct EdgeNode *next; /* 链域,指向下一个邻接点 */
    27. }EdgeNode;
    28. typedef struct VertexNode /* 顶点表结点 */
    29. {
    30. int in; /* 顶点入度 */
    31. int data; /* 顶点域,存储顶点信息 */
    32. EdgeNode *firstedge;/* 边表头指针 */
    33. }VertexNode, AdjList[MAXVEX];
    34. typedef struct
    35. {
    36. AdjList adjList;
    37. int numVertexes,numEdges; /* 图中当前顶点数和边数 */
    38. }graphAdjList,*GraphAdjList;
    39. /* **************************** */
    40. void CreateMGraph(MGraph *G)/* 构件图 */
    41. {
    42. int i, j;
    43. /* printf("请输入边数和顶点数:"); */
    44. G->numEdges=MAXEDGE;
    45. G->numVertexes=MAXVEX;
    46. for (i = 0; i < G->numVertexes; i++)/* 初始化图 */
    47. {
    48. G->vexs[i]=i;
    49. }
    50. for (i = 0; i < G->numVertexes; i++)/* 初始化图 */
    51. {
    52. for ( j = 0; j < G->numVertexes; j++)
    53. {
    54. G->arc[i][j]=0;
    55. }
    56. }
    57. G->arc[0][4]=1;
    58. G->arc[0][5]=1;
    59. G->arc[0][11]=1;
    60. G->arc[1][2]=1;
    61. G->arc[1][4]=1;
    62. G->arc[1][8]=1;
    63. G->arc[2][5]=1;
    64. G->arc[2][6]=1;
    65. G->arc[2][9]=1;
    66. G->arc[3][2]=1;
    67. G->arc[3][13]=1;
    68. G->arc[4][7]=1;
    69. G->arc[5][8]=1;
    70. G->arc[5][12]=1;
    71. G->arc[6][5]=1;
    72. G->arc[8][7]=1;
    73. G->arc[9][10]=1;
    74. G->arc[9][11]=1;
    75. G->arc[10][13]=1;
    76. G->arc[12][9]=1;
    77. }
    78. /* 利用邻接矩阵构建邻接表 */
    79. void CreateALGraph(MGraph G,GraphAdjList *GL)
    80. {
    81. int i,j;
    82. EdgeNode *e;
    83. *GL = (GraphAdjList)malloc(sizeof(graphAdjList));
    84. (*GL)->numVertexes=G.numVertexes;
    85. (*GL)->numEdges=G.numEdges;
    86. for(i= 0;i /* 读入顶点信息,建立顶点表 */
    87. {
    88. (*GL)->adjList[i].in=0;
    89. (*GL)->adjList[i].data=G.vexs[i];
    90. (*GL)->adjList[i].firstedge=NULL; /* 将边表置为空表 */
    91. }
    92. for(i=0;i/* 建立边表 */
    93. {
    94. for(j=0;j
    95. {
    96. if (G.arc[i][j]==1)
    97. {
    98. e=(EdgeNode *)malloc(sizeof(EdgeNode));
    99. e->adjvex=j; /* 邻接序号为j */
    100. e->next=(*GL)->adjList[i].firstedge; /* 将当前顶点上的指向的结点指针赋值给e */
    101. (*GL)->adjList[i].firstedge=e; /* 将当前顶点的指针指向e */
    102. (*GL)->adjList[j].in++;
    103. }
    104. }
    105. }
    106. }
    107. /* 拓扑排序,若GL无回路,则输出拓扑排序序列并返回1,若有回路返回0。 */
    108. Status TopologicalSort(GraphAdjList GL)
    109. {
    110. EdgeNode *e;
    111. int i,k,gettop;
    112. int top=0; /* 用于栈指针下标 */
    113. int count=0;/* 用于统计输出顶点的个数 */
    114. int *stack; /* 建栈将入度为0的顶点入栈 */
    115. stack=(int *)malloc(GL->numVertexes * sizeof(int) );
    116. for(i = 0; inumVertexes; i++)
    117. if(0 == GL->adjList[i].in) /* 将入度为0的顶点入栈 */
    118. stack[++top]=i;
    119. while(top!=0)
    120. {
    121. gettop=stack[top--];
    122. printf("%d -> ",GL->adjList[gettop].data);
    123. count++; /* 输出i号顶点,并计数 */
    124. for(e = GL->adjList[gettop].firstedge; e; e = e->next)
    125. {
    126. k=e->adjvex;
    127. if( !(--GL->adjList[k].in) ) /* 将i号顶点的邻接点的入度减1,如果减1后为0,则入栈 */
    128. stack[++top]=k;
    129. }
    130. }
    131. printf("\n");
    132. if(count < GL->numVertexes)
    133. return ERROR;
    134. else
    135. return OK;
    136. }
    137. int main(void)
    138. {
    139. MGraph G;
    140. GraphAdjList GL;
    141. int result;
    142. CreateMGraph(&G);
    143. CreateALGraph(G,&GL);
    144. result=TopologicalSort(GL);
    145. printf("result:%d",result);
    146. getchar();
    147. return 0;
    148. }

    AOE 网与关键路径

    AOE 网

    用顶点表示事件,用弧表示活动,权值表示活动的持续时间,这样的有向图即为 AOE 网。

    在 AOE 网中,只有一个入度为 0 的点表示工程的开始,即源点,也只有一个出度为 0 的点表示工程的结束,即汇点

    关键路径

    在 AOE 网上,从源点到汇点的具最大路径长度(该路径上行各活动持续时间的和)的路径称为关键路径,关键路径上的活动称为关键活动。

  • 相关阅读:
    JavaSE之数组
    模拟登陆系统
    医院陪诊小程序源码 医院陪诊陪护系统源码
    芒果改进YOLOv7系列:首发改进特征融合网络BiFPN结构,融合更多有效特征
    【linux下centos7.9安装docker,docker-composed(root用户)】
    跟我学c++初级篇——别名的使用
    c++ fstream 文件追加模式
    Python使用SQLAlchemy操作sqlite
    U-Net: Convolutional Networks for Biomedical Images Segmentation
    RTT学习笔记11- gpio使用和GPIO设备驱动框架层
  • 原文地址:https://blog.csdn.net/baidu_16370559/article/details/126059777