• linkedu视频
  • 平面设计
  • 电脑入门
  • 操作系统
  • 办公应用
  • 电脑硬件
  • 动画设计
  • 3D设计
  • 网页设计
  • CAD设计
  • 影音处理
  • 数据库
  • 程序设计
  • 认证考试
  • 信息管理
  • 信息安全
菜单
linkedu.com
  • 网页制作
  • 数据库
  • 程序设计
  • 操作系统
  • CMS教程
  • 游戏攻略
  • 脚本语言
  • 平面设计
  • 软件教程
  • 网络安全
  • 电脑知识
  • 服务器
  • 视频教程
  • JavaScript
  • ASP.NET
  • PHP
  • 正则表达式
  • AJAX
  • JSP
  • ASP
  • Flex
  • XML
  • 编程技巧
  • Android
  • swift
  • C#教程
  • vb
  • vb.net
  • C语言
  • Java
  • Delphi
  • 易语言
  • vc/mfc
  • 嵌入式开发
  • 游戏开发
  • ios
  • 编程问答
  • 汇编语言
  • 微信小程序
  • 数据结构
  • OpenGL
  • 架构设计
  • qt
  • 微信公众号
您的位置:首页 > 程序设计 >C语言 > C++深度优先搜索的实现方法

C++深度优先搜索的实现方法

作者: 字体:[增加 减小] 来源:互联网 时间:2017-05-28

通过本文主要向大家介绍了c++深度优先搜索,深度优先搜索方法,深度优先搜索,深度优先搜索算法,图的深度优先搜索等相关知识,希望对您有所帮助,也希望大家支持linkedu.com www.linkedu.com

本文实例讲述了图的遍历中深度优先搜索的C++实现方法,是一种非常重要的算法,具体实现方法如下:

首先,图的遍历是指从图中的某一个顶点出发,按照某种搜索方法沿着图中的边对图中的所有顶点访问一次且仅访问一次。注意到树是一种特殊的图,所以树的遍历实际上也可以看作是一种特殊的图的遍历。图的遍历主要有两种算法:广度优先搜索(Breadth-First-Search)和深度优先搜索(Depth-First-Search)。

一、深度优先搜索(DFS)的算法思想

深度优先搜索算法所遵循的搜索策略是尽可能“深”地搜索一个图。它的基本思想就是:首先访问图中某一起始顶点v,然后由v出发,访问与v邻接且未被访问的任一顶点w1,再访问与w1邻接且未被访问的任一顶点w2,……重复上述过程。当不能再继续向下访问时,依次退回到最近被访问的顶点,若它还有邻接顶点未被访问过,则从该点开始继续上述搜索过程,直到图中所有顶点均被访问过为止。

如上图所示,从顶点2开始深度优先遍历图,结果为:2,0,1,3。

二、DFS算法实现

和广度优先搜索一样,为了防止顶点被多次访问,需要使用一个访问标记数组visited[]来标记顶点是否已经被访问过。

这里使用邻接表表示图。对于一个有向图,假设从给定顶点可以访问到图的所有其他顶点,则DFS递归算法的C++代码实现:

/************************************************************************* 
  > File Name: DFS.cpp 
  > Author: SongLee 
 ************************************************************************/ 
#include<iostream> 
#include<list> 
using namespace std; 
 
/* 图 */ 
class Graph 
{ 
  int V;                // 顶点数 
  list<int> *adj;           // 邻接表 
  void DFSUtil(int v, bool visited[]); // 从顶点v深度优先遍历 
public: 
  Graph(int V);            // 构造函数 
  void addEdge(int v, int w);     // 向图中添加边 
  void DFS(int v);           // 从v开始深度优先遍历图 
}; 
 
/* 构造函数 */ 
Graph::Graph(int V) 
{ 
  this->V = V; 
  adj = new list<int>[V]; 
} 
 
/* 添加边,构造邻接表 */ 
void Graph::addEdge(int v, int w) 
{ 
  adj[v].push_back(w);         // 将w添加到v的链表 
} 
 
/* 从v开始深度优先遍历 */ 
void Graph::DFSUtil(int v, bool visited[]) 
{ 
  // 访问顶点v并输出 
  visited[v] = true; 
  cout << v << " "; 
 
  list<int>::iterator i; 
 
  for(i=adj[v].begin(); i!=adj[v].end(); ++i) 
    if(!visited[*i])       // 若邻接点尚未访问 
      DFSUtil(*i, visited);   // 递归 
} 
 
/* 对图进行深度优先遍历,调用递归函数DFSUtil() */ 
void Graph::DFS(int v) 
{ 
  bool *visited = new bool[V]; 
  for(int i=0; i<V; ++i) 
    visited[i] = false; 
 
  // 假设从给定顶点v可以到达图的所有顶点 
  DFSUtil(v, visited); 
} 
 
/* 测试 */ 
int main() 
{ 
  Graph g(4); 
  g.addEdge(0, 1); 
  g.addEdge(0, 2); 
  g.addEdge(1, 2); 
  g.addEdge(2, 0); 
  g.addEdge(2, 3); 
  g.addEdge(3, 3); 
 
  cout << "Depth First Traversal (starting from vertex 2) \n"; 
  g.DFS(2); 
  cout << endl; 
   
  return 0; 
}

</div>

上面的代码是假设从给定顶点可以访问到图的所有其他顶点。如果没有这个假设,为了对图作一个完整的深度优先遍历,我们需要对每个顶点调用DFSUtil()。当然那之前需要先检查顶点是否已经访问过。所以我们只需要修改DFS()函数部分:

void Graph::DFS() 
{ 
  bool *visited = new bool[V]; 
  for(int i=0; i<V; ++i) 
    visited[i] = false; 
   
  // 对每个顶点调用DFSUtil(),从0开始 
  for(int i=0; i<V; ++i) 
    if(!visited[i]) 
      DFSUtil(i, visited); 
} 
</div>

对于无向图的深度优先搜索,只是邻接表不一样,其他的都是一样的。我们只需要修改addEdge(v, w)函数:

void Graph::addEdge(int v, int w) 
{ 
  adj[v].push_back(w);     // 将w加到v的list 
  adj[w].push_back(v); 
} 
</div>

注意:图的邻接矩阵表示是唯一的,但对于邻接表来说,如果边的输入次序不同,生成的邻接表也不同。因此,对于同一个图,基于邻接矩阵的遍历所得到的DFS序列和BFS序列是唯一的,基于邻接表的遍历所得到的DFS序列和BFS序列是不唯一的。

三、DFS算法性能分析

1 . 空间复杂度

DFS算法是一个递归算法,需要借助一个递归工作栈,故它的空间复杂度为O(|V|)。

2 . 时间复杂度

当以邻接表存储时,时间复杂度为O(|V|+|E|)。

当以邻接矩阵存储时,时间复杂度为O(|V|^2)。

</div>
分享到:QQ空间新浪微博腾讯微博微信百度贴吧QQ好友复制网址打印

您可能想查找下面的文章:

  • 深度理解c++中的this指针
  • C++深度优先搜索的实现方法

相关文章

  • 2017-05-28C/C++宏定义的可变参数详细解析
  • 2017-05-28C语言 扫雷程序的实现
  • 2017-05-28详解C语言中的memset()函数
  • 2017-05-28用c语言实现2000内既能被3整除又能被7整除的个数
  • 2017-05-28利用C++实现矩阵的相加/相称/转置/求鞍点
  • 2017-05-28C语言 运算符详细介绍及示例代码
  • 2017-10-30Find K-th Smallest Pair Distance:查找数组元素中差值第K大的两个元素的差值
  • 2017-05-28深入分析C++中声明与定义的区别
  • 2017-05-28C语言文件操作函数freopen详细解析
  • 2017-05-28C++ 动态创建按钮及 按钮的消息响应

文章分类

  • JavaScript
  • ASP.NET
  • PHP
  • 正则表达式
  • AJAX
  • JSP
  • ASP
  • Flex
  • XML
  • 编程技巧
  • Android
  • swift
  • C#教程
  • vb
  • vb.net
  • C语言
  • Java
  • Delphi
  • 易语言
  • vc/mfc
  • 嵌入式开发
  • 游戏开发
  • ios
  • 编程问答
  • 汇编语言
  • 微信小程序
  • 数据结构
  • OpenGL
  • 架构设计
  • qt
  • 微信公众号

最近更新的内容

    • c++获取sqlite3数据库表中所有字段的方法小结
    • C++ 中实现把EXCEL的数据导入数据库(ACCESS、MSSQL等)实例代码
    • C++ 中使用lambda代替 unique_ptr 的Deleter的方法
    • C#复制和深度复制的实现方法
    • 浅谈c++中的stl中的map用法详解
    • string中c_str(),data(),copy(p,n)函数的用法总结
    • 使用C语言构建基本的二叉树数据结构
    • 数据结构之位图(bitmap)详解
    • C++语言实现线性表之链表实例
    • 双向链表插入删除基本应用介绍

关于我们 - 联系我们 - 免责声明 - 网站地图

©2020-2025 All Rights Reserved. linkedu.com 版权所有