深度优先遍历DFS用邻接表表示的图

深度优先遍历用邻接表表示的图

DFS the Adjacency List Graph

eryar@163.com

一、简介

创建了图之后,我们希望从图中某个顶点出发访遍图中其余顶点,且使每个顶点仅被访问一次。这一过程就是图的遍历(Traversing Graph)。图的遍历算法是求解图的连通性问题、拓朴排序和求解关键路径等算法的基础。

深度优先搜索(Depth First Search)是一种递归的遍历方法。其过程为从图中任意一顶点出发,访问与其相连接的未被访问的顶点。因此,遍历图的过程实质上是对每个顶点查找其邻接点的过程。

 

二、实现代码

依然使用邻接表来表示图的存储结构,深度优先遍历程序如下代码所示:


  1: //------------------------------------------------------------------------------
  2: //	Copyright (c) 2012 eryar All Rights Reserved.
  3: //
  4: //		File    : Main.cpp
  5: //		Author  : eryar@163.com
  6: //		Date    : 2012-8-25 17:11
  7: //		Version : 0.1v
  8: //
  9: //	Description : Use Adjacency List data structure to store Digraph.
 10: //
 11: //==============================================================================
 12:
 13: #include <vector>
 14: #include <string>
 15: #include <iostream>
 16: using namespace std;
 17:
 18: struct SVertexNode
 19: {
 20:     bool          bIsVisited;
 21:     string        data;
 22:     vector<int> vecLoc;
 23: };
 24:
 25: typedef struct SEdge
 26: {
 27:     int iInitialNode;
 28:
 29:     int iTerminalNode;
 30:
 31: }Edge;
 32:
 33: typedef struct SGraph
 34: {
 35:     int iVertexNum;
 36:     int iEdgeNum;
 37:     vector<SVertexNode> vecVertex;
 38: }Graph;
 39:
 40: ///////////////////////////////////////////////////////////////////////////////
 41: // Functions of Graph
 42: void    Initialize(Graph& g, int v);
 43: Edge    MakeEdge(int v, int w);
 44: void    InsertEdge(Graph& g, const Edge& e);
 45: void    ShowGraph(const Graph& g);
 46:
 47: // Use Depth First Search method to Traverse the graph.
 48: void    DepthFirstSearch(Graph& g);
 49: void    DepthFirstSearch(Graph& g, int v);
 50:
 51: ///////////////////////////////////////////////////////////////////////////////
 52: // Main function.
 53:
 54: int main(int agrc, char* argv[])
 55: {
 56:     Graph   aGraph;
 57:
 58:     // Initialize the graph.
 59:     Initialize(aGraph, 4);
 60:
 61:     // Insert some edges to make graph.
 62:     InsertEdge(aGraph, MakeEdge(0, 1));
 63:     InsertEdge(aGraph, MakeEdge(0, 2));
 64:     InsertEdge(aGraph, MakeEdge(2, 3));
 65:     InsertEdge(aGraph, MakeEdge(3, 0));
 66:
 67:     // Show the graph.
 68:     ShowGraph(aGraph);
 69:
 70:     // DFS traverse the graph.
 71:     DepthFirstSearch(aGraph);
 72:
 73:     return 0;
 74: }
 75:
 76: ///////////////////////////////////////////////////////////////////////////////
 77:
 78: /**
 79: * brief	Initialize the graph.
 80: *
 81: *       v: vertex number of the graph.
 82: */
 83: void Initialize( Graph& g, int v )
 84: {
 85:     char    szData[6];
 86:     SVertexNode node;
 87:
 88:     g.iVertexNum    = v;
 89:     g.iEdgeNum      = 0;
 90:
 91:     for (int i = 0; i < v; i++)
 92:     {
 93:         sprintf(szData, "V%d", i+1);
 94:         node.data   = szData;
 95:         node.bIsVisited = false;
 96:         g.vecVertex.push_back(node);
 97:     }
 98: }
 99:
100: /**
101: * brief	Make an edge by initial node and terminal node.
102: */
103: Edge MakeEdge( int v, int w )
104: {
105:     Edge    e;
106:
107:     e.iInitialNode  = v;
108:     e.iTerminalNode = w;
109:
110:     return e;
111: }
112:
113: /**
114: * brief	Insert an edge to the graph.
115: */
116: void InsertEdge( Graph& g, const Edge& e )
117: {
118:     g.vecVertex.at(e.iInitialNode).vecLoc.push_back(e.iTerminalNode);
119:
120:     // If the graph is Undigraph, need do something here...
121:     //g.vecVertex.at(e.iTerminalNode).vecLoc.push_back(e.iInitialNode);
122:
123:     g.iEdgeNum++;
124: }
125:
126: /**
127: * brief	Show the graph.
128: */
129: void ShowGraph( const Graph& g )
130: {
131:     cout<<"Show the graph: "<<endl;
132:
133:     for (int i = 0; i < g.iVertexNum; i++)
134:     {
135:         cout<<"Node "<<i<<"("<<g.vecVertex.at(i).data<<")";
136:
137:         for (int j = 0; j < g.vecVertex.at(i).vecLoc.size(); j++)
138:         {
139:             cout<<"->"<<g.vecVertex.at(i).vecLoc.at(j);
140:         }
141:
142:         cout<<endl;
143:     }
144: }
145:
146: void DepthFirstSearch( Graph& g )
147: {
148:     cout<<"Depth First Search the graph:"<<endl;
149:
150:     for (int i = 0; i < g.iVertexNum; i++)
151:     {
152:         if (!(g.vecVertex.at(i).bIsVisited))
153:         {
154:             DepthFirstSearch(g, i);
155:         }
156:     }
157: }
158:
159: void DepthFirstSearch(Graph& g, int v)
160: {
161:     int     iAdjacent   = 0;
162:     SVertexNode node    = g.vecVertex.at(v);
163:
164:     // Visit the vertex and mark it.
165:     cout<<g.vecVertex.at(v).data<<endl;
166:     g.vecVertex.at(v).bIsVisited = true;
167:
168:     // Visit the adjacent vertex.
169:     for (int i = 0; i < node.vecLoc.size(); i++)
170:     {
171:         iAdjacent   = node.vecLoc.at(i);
172:
173:         if (!(g.vecVertex.at(iAdjacent).bIsVisited))
174:         {
175:             DepthFirstSearch(g, iAdjacent);
176:         }
177:     }
178:
179: }
180:
181: 

三、输出结果


  1: Show the graph:
  2: Node 0(V1)->1->2
  3: Node 1(V2)
  4: Node 2(V3)->3
  5: Node 3(V4)->0
  6: Depth First Search the graph:
  7: V1
  8: V2
  9: V3
 10: V4
时间: 2024-11-06 07:40:13

深度优先遍历DFS用邻接表表示的图的相关文章

数据结构实践——操作用邻接表存储的图

本文是针对[数据结构基础系列(7):图]的实践. [项目 - 操作用邻接表存储的图] 假设图G采用邻接表存储,分别设计实现以下要求的算法: (1)输出出图G中每个顶点的出度: (2)求出图G中出度最大的一个顶点,输出该顶点编号: (3)计算图G中出度为0的顶点数: (4)判断图G中是否存在边<i,j>. 利用下图作为测试用图,输出结果. 提示:(1)分别设计函数实现算法:(2)不要全部实现完再测试,而是实现一个,测试一个:(3)请利用图算法库. [参考解答] #include <stdi

邻接表表示的图的广度优先遍历-Breadth First Search Graph

Breadth First Search Graph eryar@163.com 一.简介 广度优先遍历类似于树的按层次遍历过程. 假设从图中某顶点V出发,在访问了V之后依次访问V的各个未曾访问过的邻接顶点,然后分别从这些邻接点出发依次访问它们的邻接点,并使"先被访问的顶点的邻接点"先于"后被访问的顶点的邻接点"被访问,直到图中所有已被访问的顶点的邻接点都被访问到.到此时图中尚有未被访问的顶点,则另选图中一个未曾访问的顶点作为始点,重复上述过程,直到图中所有顶点都被

图的存储形式:邻接表

邻接表:邻接表是图的一种链式存储结构.在邻接表中,对图中每个顶点建立一个单链表,第i个单链表中的节点表示依附于顶点vi的边(对有向图是以顶点vi为尾的弧).每个结点有三个域组成,其中邻接点域指示与顶点vi邻接的点在途中的位置,链域指示下一条边或者弧的结点:数据域存储和边或者弧相关的信息,如权值等.每个链表上附设一个表头结点.在表头结点中,除了设置链域指向链表第一个结点之外,还设置有存储顶点vi的名.如下所示: 实现: /**************************************

图的深度优先遍历算法

前言 图的遍历与前面文章中的二叉树遍历还是存在很大区别的.所谓图的遍历指的是从图中的某一个顶点出发访问图中的其余顶点,并且需要保证每个顶点只被访问一次.由于图比二叉树复杂得多,所以前面二叉树的遍历算法在图中是行不通的.因为对于任意一个顶点来讲,都可能与其余的顶点发生连接.如果不对访问的顶点做一些处理,出发重复访问的几率是很高的.因此,一个基本思想是设置一个标记数组,主要用于标记已经被访问过的顶点.图的遍历算法主要有两种:深度优先遍历和广度优先遍历.本篇文章主要介绍的是深度优先遍历算法. 深度优先

邻接表无向图(三) Java详解

邻接表无向图的介绍 邻接表无向图是指通过邻接表表示的无向图. 上面的图G1包含了"A,B,C,D,E,F,G"共7个顶点,而且包含了"(A,C),(A,D),(A,F),(B,C),(C,D),(E,G),(F,G)"共7条边. 上图右边的矩阵是G1在内存中的邻接表示意图.每一个顶点都包含一条链表,该链表记录了"该顶点的邻接点的序号".例如,第2个顶点(顶点C)包含的链表所包含的节点的数据分别是"0,1,3":而这"

邻接表无向图(二) C++详解

邻接表无向图的介绍 邻接表无向图是指通过邻接表表示的无向图. 上面的图G1包含了"A,B,C,D,E,F,G"共7个顶点,而且包含了"(A,C),(A,D),(A,F),(B,C),(C,D),(E,G),(F,G)"共7条边. 上图右边的矩阵是G1在内存中的邻接表示意图.每一个顶点都包含一条链表,该链表记录了"该顶点的邻接点的序号".例如,第2个顶点(顶点C)包含的链表所包含的节点的数据分别是"0,1,3":而这"

邻接表无向图(一) C语言详解

邻接表无向图的介绍 邻接表无向图是指通过邻接表表示的无向图. 上面的图G1包含了"A,B,C,D,E,F,G"共7个顶点,而且包含了"(A,C),(A,D),(A,F),(B,C),(C,D),(E,G),(F,G)"共7条边. 上图右边的矩阵是G1在内存中的邻接表示意图.每一个顶点都包含一条链表,该链表记录了"该顶点的邻接点的序号".例如,第2个顶点(顶点C)包含的链表所包含的节点的数据分别是"0,1,3":而这"

邻接表有向图(二) C++详解

邻接表有向图的介绍 邻接表有向图是指通过邻接表表示的有向图. 上面的图G2包含了"A,B,C,D,E,F,G"共7个顶点,而且包含了"<A,B>,<B,C>,<B,E>,<B,F>,<C,E>,<D,C>,<E,B>,<E,D>,<F,G>"共9条边. 上图右边的矩阵是G2在内存中的邻接表示意图.每一个顶点都包含一条链表,该链表记录了"该顶点所对应的

邻接表有向图(一) C语言详解

邻接表有向图的介绍 邻接表有向图是指通过邻接表表示的有向图. 上面的图G2包含了"A,B,C,D,E,F,G"共7个顶点,而且包含了"<A,B>,<B,C>,<B,E>,<B,F>,<C,E>,<D,C>,<E,B>,<E,D>,<F,G>"共9条边. 上图右边的矩阵是G2在内存中的邻接表示意图.每一个顶点都包含一条链表,该链表记录了"该顶点所对应的