前言

查阅了网上许多关于通过DFS算法对有向图中所有简单回路的查找,发现有很多关于使用DFS求解有向回路中所有简单回路的帖子,(在按照节点编号情况下)但大多数仅仅寻找了编号递增的回路。又或者未对结果去重。P.S.下述有向图中所有节点均使用数字进行编号,如节点0、节点1 \(\cdots\)

1. 算法描述

本算法基于DFS,思路与传统DFS基本类似,只不过在遍历过程中对所经过的路径通过一个栈进行保存,当找到回路时,检测此条回路是否已经在结果集中出现,若未出现,则将其放入结果集。本过程中比较关键的两步是DFS结果集去重



关于DFS。每当从一个新的顶点出发对其进行递归的深度遍历时,我们维护一个新的节点已访问数组visited[]以及一个一维的回路栈loopStack。当访问到节点v时,若v已经被访问,即visited[v]==true时,扫描栈中是节点v是否已经出现过,当节点v已经在栈中出现过,则说明此时出现一条回路,我们将其加入结果集。若节点v未被访问,此时置visited[v]=true并将节点v入栈,并对v的所有邻接点进行访问,重复以上操作。到最深处(即已无邻接点未遍历)进行回溯处理,即将栈顶元素退栈。



关于去重。当需要在结果集中加入新找到的一条回路时,需要对结果集扫描,判断此条回路是否已经出现过。但在一个有向图中很容易发现回路\(0\rightarrow1\rightarrow2\rightarrow3(\rightarrow0)\)与回路\(2\rightarrow3\rightarrow1\rightarrow0(\rightarrow2)\)是两条相同的回路,那么在结果集中我们需要对此类回路进行去重,此处我的具体做法是使用两个指针i、j对将要比较的两条回路同时进行扫描比较,指针i指向第一条回路的起始位置,指针j指向第二条回路中,与指针i指向位置元素相等的位置,记录两个回路中相等的元素个数count,当count==loop.size()时,我们称这两条回路为同一条回路,否则将新回路加入结果集。

算法具体步骤:

  1. 从v出发对图进行深度遍历若此节点已访问则转2,否则转3。
  2. 若此节点已经在loopStack中出现,表明有回路存在,判断回路是否已经在结果集loopStacks中出现,若没出现过,则放入结果集。
  3. 置Visited[v]=true,节点v入栈,对v的邻接顶点继续进行深搜。当搜索完所有邻接顶点,栈顶元素退栈。

###2. 算法实现
完整代码在github,请[点击这里](https://github.com/geTiger/FindLoopsInGraph/blob/master/FindAllSimpleLoopsDFS.h)
DFS部分实现
```
void DFS(int v) {
int position = FindInVector(loopStack, v);
if (position ==-1 && visited[v])
visited[v] =false;
if (visited[v] == 1) {
if (position >=0) {
vector loop;
//将环单独拿出并放入结果集
for (int i = position; i for (int j = 0; j < adjMatrixSize; j++) {
if (adjMatrix[v][j] == 1)
DFS(j);
}

loopStack.pop_back();

}

<br>

结果集去重部分

void AddLoopStack(vector loop) {

bool haveThisLoop = false;

int count,begin;

if (loopStacks.size() == 0)

loopStacks.push_back(loop);

else {

for (int i = 0; i < loopStacks.size(); i++) {//遍历结果集中的每一个回路

count = 0;

begin = 0;

if (loop.size() == loopStacks[i].size()) {//若长度相等则进一步比较

//为方便比较,找到结果集中第i条回路中与loop进行匹配的起点

for (int k = 0; k < loopStacks[i].size(); k++) {

if (loop[0] == loopStacks[i][k])

begin = k;

}

//j指针从待添加结果集的loop数组的头部开始扫描

//k指针从上述所找出的与loop数组比较的起点开始扫描

for (int j =0,k=begin; j < loop.size(); j++, k = (k + 1) % (loopStacks[i].size())) {

if (loop[j] == loopStacks[i][k])

count++;

}

if (count == loop.size()) {

//haveThisLoop = true;

//break;

return;

}

}

}//end else

	loopStacks.push_back(loop);
}

}


###3. 算法复杂度分析
假设存在n个顶点,考虑最坏情况下,有向图为有向完全图,那么可能的回路个数就是$C_n^2+C_n^3+\cdots+C_n^n=2^n-n-1$个回路,另外需要对n个顶点均需要DFS,而每条边都需要经过,加上去重的部分,所以时间复杂度为$O(n^32^n)$,需要保存所有回路的空间,故空间复杂度为$O(2^n)$。
<br>
<hr>
参考资料
严蔚敏数据结构(C语言版)

通过DFS求解有向图(邻接表存储)中所有简单回路的更多相关文章

  1. 数据结构(11) -- 邻接表存储图的DFS和BFS

    /////////////////////////////////////////////////////////////// //图的邻接表表示法以及DFS和BFS //////////////// ...

  2. 邻接表存储图,DFS遍历图的java代码实现

    import java.util.*; public class Main{ static int MAX_VERTEXNUM = 100; static int [] visited = new i ...

  3. 数据结构之---C语言实现图的邻接表存储表示

    // 图的数组(邻接矩阵)存储表示 #include <stdio.h> #include <stdlib.h> #include <string.h> #defi ...

  4. 图的邻接表存储 c实现

    图的邻接表存储 c实现 (转载) 用到的数据结构是 一个是顶点表,包括顶点和指向下一个邻接点的指针 一个是边表, 数据结构跟顶点不同,存储的是顶点的序号,和指向下一个的指针 刚开始的时候把顶点表初始化 ...

  5. PTA 邻接表存储图的广度优先遍历(20 分)

    6-2 邻接表存储图的广度优先遍历(20 分) 试实现邻接表存储图的广度优先遍历. 函数接口定义: void BFS ( LGraph Graph, Vertex S, void (*Visit)(V ...

  6. hdu 4707 Pet(DFS &amp;&amp; 邻接表)

    Pet Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submis ...

  7. PTA 邻接表存储图的广度优先遍历

    试实现邻接表存储图的广度优先遍历. 函数接口定义: void BFS ( LGraph Graph, Vertex S, void (*Visit)(Vertex) ) 其中LGraph是邻接表存储的 ...

  8. DS实验题 Old_Driver UnionFindSet结构 指针实现邻接表存储

    题目见前文:DS实验题 Old_Driver UnionFindSet结构 这里使用邻接表存储敌人之间的关系,邻接表用指针实现: // // main.cpp // Old_Driver3 // // ...

  9. 图的邻接表存储表示(C)

    //---------图的邻接表存储表示------- #include<stdio.h> #include<stdlib.h> #define MAX_VERTEXT_NUM ...

随机推荐

  1. Delphi 中的 IfThen 函数

    问题来源: http://www.cnblogs.com/del/archive/2008/11/14/1120015.html#1370413 StrUtils 单元和 Math 单元 分别有一个 ...

  2. orcale mysql基本的分页查询法

    orcale分页查询sql语句: SELECT * FROM ( SELECT A.*, ROWNUM RN FROM (SELECT * FROM TABLE_NAME) A WHERE ROWNU ...

  3. Akka-Cluster(5)- load-balancing with backoff-supervised stateless computation - 无状态任务集群节点均衡分配

    分布式程序运算是一种水平扩展(scale-out)运算模式,其核心思想是能够充分利用服务器集群中每个服务器节点的计算资源,包括:CPU.内存.硬盘.IO总线等.首先对计算任务进行分割,然后把细分的任务 ...

  4. 知乎十万级容器规模的Java分布式镜像仓库实践

    作者:知乎令孤无忌 前言 知乎在 2016 年已经完成了全量业务的容器化,并在自研容器平台上以原生镜像的方式部署和运行,并在后续陆续实施了 CI.Cron.Kafka.HAProxy.HBase.Tw ...

  5. Javascript高级编程学习笔记(45)——DOM 操作表格及DOM动态集合

    操作DOM表格 早些时候,HTML 还是以表格布局为主, 所以DOM操作表格是比较重要的一点 但是现如今 有其它的选择,所以表格的操作也就慢慢地淡出了人们的视线 所以这里也就不过多去详细展开,这里也就 ...

  6. Javascript高级编程学习笔记(17)—— 引用类型(6)基本包装类

    基本包装类 基本包装类这个概念或许有的小伙伴没有听说过 但是小伙伴们有没有想过,为什么基本数据类型的实例也有方法呢? 其实这些方法都来自基本包装类型 这是JS为了方便操作基础数据类型而创建的特殊引用类 ...

  7. Java项目启动时执行指定方法的几种方式

    很多时候我们都会碰到需要在程序启动时去执行的方法,比如说去读取某个配置,预加载缓存,定时任务的初始化等.这里给出几种解决方案供大家参考. 1. 使用@PostConstruct注解 这个注解呢,可以在 ...

  8. 每天学点SpringCloud(三):自定义Eureka集群负载均衡策略

    相信看了 每天学点SpringCloud(一):简单服务提供者消费者调用,每天学点SpringCloud(二):服务注册与发现Eureka这两篇的同学都了解到了我的套路,没错,本篇博客同样是为了解决上 ...

  9. [Postman]捕获HTTP请求(14)

    如果您使用API​​构建客户端应用程序 - 移动应用程序,网站或桌面应用程序 - 您可能希望查看应用程序中发送和接收的实际HTTP请求流量.在某些情况下,您可能会发现甚至没有记录的API.Postma ...

  10. npm包实现发布正式和测试版

    npm publish的時候 怎麽發測試版和正式版本呢? 通常我們一般情況下 直接 npm publish 提交自己的開發包后,在項目中 npm install @packageName 是下載下來剛 ...