转载请注明出处:http://blog.csdn.net/ns_code/article/details/19617187

图的存储结构

本文的重点在于图的深度优先搜索(DFS)和广度优先搜索(BFS),因此不再对图的基本概念做过多的介绍,但是要先大致了解下图的几种常见的存储结构。

邻接矩阵

邻接矩阵既可以用来存储无向图,也可以用来存储有向图。该结构实际上就是用一个二维数组(邻接矩阵)来存储顶点的信息和顶点之间的关系(有向图的弧或无向图的边)。其描述形式如下:

  1. //图的邻接矩阵存储表示
  2. #define MAX_NUM 20 // 最大顶点个数
  3. enum GraphKind{GY,GN}; // {有向图,无向图}
  4. typedef struct
  5. {
  6. VRType adj; // 顶点关系类型。对无权图,用1(是)或0(否)表示是否相邻;对带权图,则为权值
  7. InfoType *info; // 与该弧或边相关信息的指针(可无)
  8. }ArcCell,AdjMatrix[MAX_NUM][MAX_NUM]; // 二维数组
  9. typedef struct
  10. {
  11. VertexType vexs[MAX_NUM]; // 顶点向量
  12. AdjMatrix arcs; // 邻接矩阵
  13. int vexnum,arcnum; // 图的当前顶点数和弧(边)数
  14. GraphKind kind; // 图的种类标志
  15. }Graph;

我们分别看下面两个图,左边为有向图,右边为无向图

             
    上面两个图均为无权图,我们假设存储的时候,V0的序号为0,V1的序号为1,V2的序号为2。。。,且adj为1表示两顶点间没有没有连接,为0时表示有连接。则有向图的邻接矩阵如下图左边的矩阵所示,无向图的邻接矩阵如下图右边的矩阵所示;
 
       
    根据邻接矩阵很容易判断图中任意两个顶点之间连通与否,并可以求出各个顶点的度。
    1、对于无向图,观察右边的矩阵,发现顶点Vi的度即是邻接矩阵中第i行(或第i列)的元素之和。
    2、对于有向图,由于需要分别计算出度和入读,观察左边的矩阵,发现顶点Vi的出度即为邻接矩阵第i行元素之和,入度即为邻接矩阵第i列元素之和,因此顶点Vi的度即为邻接矩阵中第i行元素和第i列元素之和。
    很明显,邻接矩阵所占用的存储空间与图的边数或弧数无关,因此适用于边数或弧数较多的稠密图。


邻接表

邻接表是图的一种链式存储结构,既适合于存储无向图,也适合于存储有向图。在邻接表中,用一个一维数组存储图中的每个顶点的信息,同时为每个顶点建立一个单链表,链表中的节点保存依附在该顶点上的边或弧的信息。其描述形式如下:

  1. //图的邻接表存储表示
  2. #define MAX_NUM 20
  3. enum GraphKind{GY,GN}; // {有向图,无向图}
  4. typedef struct
  5. {
  6. int adjvex; // 该弧所指向的顶点或边的另一个顶点的位置
  7. ArcNode *nextarc; // 指向下一条弧或边的指针
  8. InfoType *info; // 与弧或边相关信息的指针(可无)
  9. }ArcNode;// 表结点
  10. typedef struct
  11. {
  12. VertexType data; // 顶点信息
  13. ArcNode *firstarc; // 第一个表结点的地址,指向第一条依附该顶点的弧或边的指针
  14. }VNode,AdjList[MAX_NUM]; // 头结点
  15. struct
  16. {
  17. AdjList vertices;
  18. int vexnum,arcnum; // 图的当前顶点数和弧(边)数
  19. GraphKind kind; // 图的种类标志
  20. }Graph;
    依然以上面的有向图和无向图为例,采用邻接表来存储,可以得到对应的存储形式如下:
    在邻接表上容易找到任意一个顶点的第一个邻接点和下一个邻接点,但是要判定任意两个顶点之间是否有边或弧需搜索两个顶点对应的链表,不及邻接矩阵方便。
    很明显,邻接表所占用的存储空间与图的边数或弧数有关,因此适用于边数或弧数较少的稀疏图。


十字链表

十字链表也是一种链式存储结构,用来表示有向图,它在有向图邻接表的基础上加入了指向弧尾的指针。表示形式如下:

  1. //有向图的十字链表存储表示
  2. #define MAX_NUM 20
  3. typedef struct // 弧结点
  4. {
  5. int tailvex,headvex; // 该弧的尾和头顶点的位置
  6. ArcBox *hlink,*tlink; // 分别为弧头相同和弧尾相同的弧的链域
  7. InfoType *info; // 该弧相关信息的指针,可指向权值或其他信息(可无)
  8. }ArcBox;
  9. typedef struct // 顶点结点
  10. {
  11. VertexType data;
  12. ArcBox *firstin,*firstout; // 分别指向该顶点第一条入弧和出弧
  13. }VexNode;
  14. struct
  15. {
  16. VexNode xlist[MAX_NUM]; // 表头向量(数组)
  17. int vexnum,arcnum; // 有向图的当前顶点数和弧数
  18. }Graph;
    其思想也很容易理解,这里不再细说。
    在十字链表中,既容易找到以某个顶点为尾的弧,也容易找到以某个顶点为头的弧,因而容易求得顶点的出度和入度。

邻接多重表

邻接多重表也是一种链式存储结构,用来表示无向图,与有向图的十字链表相似,这里也不再细说,直接给出其表示形式:

  1. //无向图的邻接多重表存储表示
  2. #define MAX_NUM 20
  3. typedef struct
  4. {
  5. VisitIf mark; // 访问标记
  6. int ivex,jvex; // 该边依附的两个顶点的位置
  7. EBox *ilink,*jlink; // 分别指向依附这两个顶点的下一条边
  8. InfoType *info; // 该边信息指针,可指向权值或其他信息
  9. }EBox;
  10. typedef struct
  11. {
  12. VertexType data;
  13. EBox *firstedge; // 指向第一条依附该顶点的边
  14. }VexBox;
  15. typedef struct
  16. {
  17. VexBox adjmulist[MAX_NUM];
  18. int vexnum,edgenum; // 无向图的当前顶点数和边数
  19. }Graph;

图的遍历

图的遍历比树的遍历要复杂的多,因为图的任一顶点都有可能与其他的顶点相邻接,因此,我们需要设置一个辅助数组visited[]来标记某个节点是否已经被访问过。图的遍历通常有两种方法:深度优先遍历(BFS)和广度优先遍历(DFS),两种遍历方式的思想都不难理解。下面我们就以如下图所示的例子来说明图的这两种遍历方式的基本思想,并用邻接表作为图的存储结构,给出BFS和DFS的实现代码(下图为无向图,有向图的BFS和DFS实现代码与无向图的相同):

我们以邻接表作为上图的存储结构,并将A、B、C、D、E、F、G、H在顶点数组中的序号分别设为0、1、2、3、4、5、6、7。我们根据上图所包含的信息,精简了邻接表的存储结构,采取如下所示的结构来存储图中顶点和边的相关信息:

  1. #define NUM 8          //图中顶点的个数
  2. /*
  3. 用邻接表作为图的存储结构
  4. 在邻接表中,用一个一维数组存储图中的每个顶点的信息,
  5. 同时为每个顶点建立一个单链表,链表中的节点保存依附在该顶点上的边或弧的信息
  6. */
  7. typedef struct node
  8. {   //链表中的每个节点,保存依附在该节点上的边或弧的信息
  9. int vertex;          //在有向图中表示该弧所指向的顶点(即弧头)的位置,
  10. //在无向图中表示依附在该边上的另一个顶点的位置
  11. struct node *pNext;  //指向下一条依附在该顶点上的弧或边
  12. }Node;
  13. typedef struct head
  14. {   //数组中的每个元素,保存图中每个顶点的相关信息
  15. char data;          //顶点的数据域
  16. Node *first;        //在有向图中,指向以该顶点为弧尾的第一条弧
  17. //在无向图中,指向依附在该顶点上的第一条边
  18. }Head,*Graph;           //动态分配数组保存每个顶点的相关信息

那么用邻接表来表示上图中各顶点间的关系,如下图所示:

深度优先搜索

深度优先搜索(DFS)类似于树的先序遍历(如尚未掌握图的先序遍历,请移步这里:http://blog.csdn.net/ns_code/article/details/12977901)。其基本思想是:从图中某个顶点v出发,遍历该顶点,而后依次从v的未被访问的邻接点出发深度优先遍历图中,直到图中所有与v连通的顶点都已被遍历。如果当前图只是需要遍历的非连通图的一个极大连通子图,则另选其他子图中的一个顶点,重复上述遍历过程,直到该非连通图中的所有顶点都被遍历完。很明显,这里要用到递归的思想。

结合上面的例子来分析,假设从A点开始遍历该图,根据图中顶点的存储关系,会按照下面的步骤来遍历该图:

1、在访问完顶点A后,选择A的第一个邻接点B进行访问;

2、而后看B的邻接点,由于B的第一个邻接点A已经被访问过,故选择其下一个邻接点D进行访问;

3、而后看D的邻接点,由于D的第一个邻接点B已经被访问过,故选择其下一个邻接点H进行访问;

4、而后看H的邻接点,由于H的第一个邻接点D已经被访问过,故选择其下一个邻接点E进行访问;

5、而后看E的邻接点,由于E的第一个邻接点B已经被访问过,那么看其第二个邻接点H,也被访问过了,E的邻接点已经全部被访问过了。

6、这时退回到上一层递归,回到顶点H,同样H的邻接点也都被访问完毕,再退回到D,D的邻接点也已访问完毕,再退回到B,一直退回到A;

7、由于A的下一个邻接点C还没有被访问,因此访问C;

8、而后看C的邻接点,由于C的第一个邻接点A已经被访问过,故选择其下一个邻接点F进行访问;

9、而后看F的邻接点,由于F的第一个邻接点C已经被访问过,故选择其下一个邻接点G进行访问;

10、而后看G的邻接点,由于G的临界点都已被访问完毕,因此会退到上一层递归,看F顶点;

11、同理,由F再回退到C,再由C回退到A,这是第一层的递归,A的所有邻接点也已都被访问完毕,遍历到此结束。

综上,上图的DFS是按照如下顺序进行的:

A->B->D->H->E->H->D->B->A->C->F->G->F->C->A

其中,红色部分代表首次访问,在这部分顶点被访问后,我们便将visited数组的对应元素设为true,表明这些顶点已经被访问过了,因此我们可以得到深度优先搜索得到的顺序为:

A->B->D->H->E->C->F->G

深度优先搜索的代码实现如下:

  1. /*
  2. 从序号为begin的顶点出发,递归深度优先遍历连通图Gp
  3. */
  4. void DFS(Graph Gp, int begin)
  5. {
  6. //遍历输出序号为begin的顶点的数据域,并保存遍历信息
  7. printf("%c ",Gp[begin].data);
  8. visited[begin] = true;
  9. //循环访问当前节点的所有邻接点(即该节点对应的链表)
  10. int i;
  11. for(i=first_vertex(Gp,begin); i>=0; i=next_vertex(Gp,begin,i))
  12. {
  13. if(!visited[i])  //对于尚未遍历的邻接节点,递归调用DFS
  14. DFS(Gp,i);
  15. }
  16. }
  17. /*
  18. 从序号为begin的节点开始深度优先遍历图Gp,Gp可以是连通图也可以是非连通图
  19. */
  20. void DFS_traverse(Graph Gp,int begin)
  21. {
  22. int i;
  23. for(i=0;i<NUM;i++)    //初始化遍历标志数组
  24. visited[i] = false;
  25. //先从序号为begin的顶点开始遍历对应的连通图
  26. DFS(Gp,begin);
  27. //如果是非连通图,该循环保证每个极大连通子图中的顶点都能被遍历到
  28. for(i=0;i<NUM;i++)
  29. {
  30. if(!visited[i])
  31. DFS(Gp,i);
  32. }
  33. }

这里调用了first_vertex()和next_vertex()两个函数,根据如上图手绘的邻接表的存储结构,这两个函数代码实现及详细注释如下:

  1. /*
  2. 返回图Gp中pos顶点(序号为pos的顶点)的第一个邻接顶点的序号,如果不存在,则返回-1
  3. */
  4. int first_vertex(Graph Gp,int pos)
  5. {
  6. if(Gp[pos].first)  //如果存在邻接顶点,返回第一个邻接顶点的序号
  7. return  Gp[pos].first->vertex;
  8. else              //如果不存在,则返回-1
  9. return -1;
  10. }
  11. /*
  12. cur顶点是pos顶点(cur和pos均为顶点的序号)的其中一个邻接顶点,
  13. 返回图Gp中,pos顶点的(相对于cur顶点)下一个邻接顶点的序号,如果不存在,则返回-1
  14. */
  15. int next_vertex(Graph Gp,int pos,int cur)
  16. {
  17. Node *p = Gp[pos].first; //p初始指向顶点的第一个邻接点
  18. //循环pos节点对应的链表,直到p指向序号为cur的邻接点
  19. while(p->vertex != cur)
  20. p = p->pNext;
  21. //返回下一个节点的序号
  22. if(p->pNext)
  23. return p->pNext->vertex;
  24. else
  25. return -1;
  26. }

广度优先搜索

广度优先搜索(BFS)类似于树的层序遍历(如尚未掌握图的先序遍历,请移步这里:http://blog.csdn.net/ns_code/article/details/13169703)。其基本思想是:从头图中某个顶点v出发,访问v之后,依次访问v的各个未被访问的邻接点,而后再分别从这些邻接点出发,依次访问它们的邻接点,直到图中所有与v连通的顶点都已被访问。如果当前图只是需要遍历的非连通图的一个极大连通子图,则另选其他子图中的一个顶点,重复上述遍历过程,直到该非连通图中的所有顶点都被遍历完。很明显,跟树的层序遍历一样,图的广度优先搜索要用到队列来辅助实现。

同样以上面的例子来分析,假设从A点开始遍历该图,根据图中顶点的存储关系,会按照下面的步骤来遍历该图:

1、在访问完顶点A后,首先访问A的第一个邻接点B,而后访问A的第二个邻接点C;

2、再根据顺序访问B的未被访问的邻接点,先访问D,后访问E;

3、再根据顺序访问C的未被访问的邻接点,先访问F,在访问G;

4、而后一次访问D、E、F、G等顶点的未被访问的邻接点;

5、D的邻接点中,只有H未被访问,因地访问H;

6、E、F、G、H等都没有未被访问的邻接点了,遍历到此结束。

综上,我们可以得到广度优先搜索得到的顺序为:

A->B->C->D->E->F->G->H

深度优先搜索的代码实现如下:

  1. /*
  2. 从序号为begin的顶点开始,广度优先遍历图Gp,Gp可以是连通图也可以是非连通图
  3. */
  4. void BFS_traverse(Graph Gp,int begin)
  5. {
  6. int i;
  7. for(i=0;i<NUM;i++)    //初始化遍历标志数组
  8. visited[i] = false;
  9. //先从序号为begin的顶点开始遍历对应的连通图
  10. BFS(Gp,begin);
  11. //如果是非连通图,该循环保证每个极大连通子图中的顶点都能被遍历到
  12. for(i=0;i<NUM;i++)
  13. {   if(!visited[i])
  14. BFS(Gp,i);
  15. }
  16. }
  17. /*
  18. 从序号为begin的顶点开始,广度优先遍历连通图Gp
  19. */
  20. void BFS(Graph Gp,int begin)
  21. {
  22. //遍历输出序号为begin的顶点的数据域,并保存遍历信息
  23. printf("%c ",Gp[begin].data);
  24. visited[begin] = true;
  25. int i;
  26. int pVertex;  //用来保存从队列中出队的顶点的序号
  27. PQUEUE queue = create_queue();  //创建一个空的辅助队列
  28. en_queue(queue, begin);          //首先将序号为begin的顶点入队
  29. while(!is_empty(queue))
  30. {
  31. de_queue(queue,&pVertex);
  32. //循环遍历,访问完pVertex顶点的所有邻接顶点,并将访问后的邻接顶点入队
  33. for(i=first_vertex(Gp,pVertex); i>=0; i=next_vertex(Gp,pVertex,i))
  34. {
  35. if(!visited[i])
  36. {
  37. printf("%c ",Gp[i].data);
  38. visited[i] = true;
  39. en_queue(queue,i);
  40. }
  41. }
  42. }
  43. //销毁队列,释放其对应的内存
  44. destroy_queue(queue);
  45. }

遍历结果

按照上面的例子来构建图,而后采用如下代码测试DFS和BFS的输出结果:

  1. /**********************************
  2. 图的BFS和DFS
  3. Author:兰亭风雨 Date:2014-02-20
  4. Email:zyb_maodun@163.com
  5. **********************************/
  6. #include<stdio.h>
  7. #include<stdlib.h>
  8. #include "data_structure.h"
  9. int main()
  10. {
  11. Graph Gp = create_graph();
  12. //深度优先遍历
  13. printf("对图进行深度优先遍历:\n");
  14. printf("从顶点A出发DFS的结果:");
  15. DFS_traverse(Gp,0);
  16. printf("\n");
  17. printf("从顶点H出发DFS的结果:");
  18. DFS_traverse(Gp,7);
  19. printf("\n");
  20. printf("从顶点E出发DFS的结果:");
  21. DFS_traverse(Gp,4);
  22. printf("\n");
  23. printf("\n");
  24. //广度优先遍历
  25. printf("对图进行深度优先遍历:\n");
  26. printf("从顶点A出发BFS的结果:");
  27. BFS_traverse(Gp,0);
  28. printf("\n");
  29. printf("从顶点H出发BFS的结果:");
  30. BFS_traverse(Gp,7);
  31. printf("\n");
  32. printf("从顶点E出发BFS的结果:");
  33. BFS_traverse(Gp,4);
  34. printf("\n");
  35. int i;
  36. //释放掉为每个单链表所分配的内存
  37. for(i=0;i<NUM;i++)
  38. {
  39. free(Gp[i].first);
  40. Gp[i].first = 0;  //防止悬垂指针的产生
  41. }
  42. //释放掉为顶点数组所分配的内存
  43. free(Gp);
  44. Gp = 0;
  45. return 0;
  46. }

测试得到的输出结果如下:

完整代码下载

    完整的C语言实现代码下载地址:http://download.csdn.net/detail/mmc_maodun/6946859


【数据结构与算法】自己动手实现图的BFS和DFS(附完整源码)的更多相关文章

  1. 数据结构与算法系列2 线性表 使用java实现动态数组+ArrayList源码详解

    数据结构与算法系列2 线性表 使用java实现动态数组+ArrayList源码详解 对数组有不了解的可以先看看我的另一篇文章,那篇文章对数组有很多详细的解析,而本篇文章则着重讲动态数组,另一篇文章链接 ...

  2. 微信小程序多图上传/朋友圈传图效果【附完整源码】

    效果图 部分源代码 js文件: var uploadPicture = require('../Frameworks/common.js') //获取应用实例 const app = getApp() ...

  3. Web思维导图实现的技术点分析(附完整源码)

    简介 思维导图是一种常见的表达发散性思维的有效工具,市面上有非常多的工具可以用来画思维导图,有免费的也有收费的,此外也有一些可以用来帮助快速实现的JavaScript类库,如:jsMind.KityM ...

  4. [算法2-数组与字符串的查找与匹配] (.NET源码学习)

    [算法2-数组与字符串的查找与匹配] (.NET源码学习) 关键词:1. 数组查找(算法)   2. 字符串查找(算法)   3. C#中的String(源码)   4. 特性Attribute 与内 ...

  5. 数据结构与算法系列研究七——图、prim算法、dijkstra算法

    图.prim算法.dijkstra算法 1. 图的定义 图(Graph)可以简单表示为G=<V, E>,其中V称为顶点(vertex)集合,E称为边(edge)集合.图论中的图(graph ...

  6. 数据结构(三十二)图的遍历(DFS、BFS)

    图的遍历和树的遍历类似.图的遍历是指从图中的某个顶点出发,对图中的所有顶点访问且仅访问一次的过程.通常有两种遍历次序方案:深度优先遍历和广度优先遍历. 一.深度优先遍历 深度优先遍历(Depth_Fi ...

  7. 多图详解Go的sync.Pool源码

    转载请声明出处哦~,本篇文章发布于luozhiyun的博客:https://www.luozhiyun.com 本文使用的go的源码时14.4 Pool介绍 总所周知Go 是一个自动垃圾回收的编程语言 ...

  8. java实现插入排序算法 附单元测试源码

    插入排序算法 public class InsertSortTest { /** * @param args */ public static void main(String[] args) { i ...

  9. 多图详解Go中的Channel源码

    转载请声明出处哦~,本篇文章发布于luozhiyun的博客:https://www.luozhiyun.com 本文使用的go的源码时14.4 chan介绍 package main import & ...

随机推荐

  1. 用secureCRT操作ubuntu终端

    用secureCRT操作ubuntu终端 ubuntu下先安装ssh windows下win+R再输入ubuntu的ip地址   ubuntu 检测端口号的命令 netstat -antp   下载到 ...

  2. sqlserver中将查询结果拼接成字符串

    #for xml path(param)--将查询结果以xml格式输出 select id,name from table1 for xml path --id和name为table1的真实字段 - ...

  3. SQLServer语言之DDL,DML,DCL,TCL

    数据库语言分类 SQLServer   SQL主要分成四部分: (1)数据定义.(SQL DDL)用于定义SQL模式.基本表.视图和索引的创建和撤消操作. (2)数据操纵.(SQL DML)数据操纵分 ...

  4. Visual Studio 2017中的快捷键

    Ctrl+Tab: 快速切换活动文件

  5. unbuntu中如何像Windows一样顺畅的切换中英文输入法

    1.首先在unbuntu安装搜狗拼音输入法(这个不用教了) 2.点击右上角的搜狗拼音的图标点击设置进入设置页面 3.选择高级 4.选择Fcitx设置 5.添加输入法英语(美国) 6.在设置中选择按键, ...

  6. linux系统编译安装软件的通用步骤

    编译安装的步骤: 1.下载源代码,并解压     tar -xf package-version.tar.{gz|bz2|xz} 注意:展开后的目录通常为package-version 2.切换至源码 ...

  7. 并发之线程封闭与ThreadLocal解析

    并发之线程封闭与ThreadLocal解析 什么是线程封闭 实现一个好的并发并非易事,最好的并发代码就是尽量避免并发.而避免并发的最好办法就是线程封闭,那什么是线程封闭呢? 线程封闭(thread c ...

  8. Python 索引迭代

    1.使用enumerate函数 L = ['Adam', 'Lisa', 'Bart', 'Paul'] for index, name in enumerate(L):     print inde ...

  9. NOI 2012 随机数生成器

    看到全是矩阵的题解,我来一发递推+分治 其实这题一半和poj1845很像(或是1875?一个叫Sumdiv的题) 言归正传,我们看看怎么由f(0)推出f(n) 我们发现,题目中给出了f(n)=af(n ...

  10. python 全栈开发,Day124(MongoDB初识,增删改查操作,数据类型,$关键字以及$修改器,"$"的奇妙用法,Array Object 的特殊操作,选取跳过排序,客户端操作)

    一.MongoDB初识 什么是MongoDB MongoDB 是一个基于分布式文件存储的数据库.由 C++ 语言编写.旨在为 WEB 应用提供可扩展的高性能数据存储解决方案. MongoDB 是一个介 ...