C++ 树进阶系列之探讨深度搜索算法查找环的细枝末节

news2025/1/24 2:17:49

1. 前言

对于基环树的讲解,分上、下 2 篇,上篇以理解连通分量、环以及使用深度搜索算法检查连通性和环为主,下篇以基于基环树结构的应用为主。

什么是基环树?

所谓基环树指由n个节点n条边所构建而成的连通图

如下图所示,树结构中共有 7 个节点, 6 条边。此时在树结构上添加一条边,必然会形成一个树环。因树结构中有环,故得此名。基环树也称为环套树。

1.png

如下图基环树结构中有 7 个节点,7 条边。

2.png

上述为无向边基环树。针对于有向边,基环树分:

  • 内向树:树中每个点有且仅有一条出边(或者说每个节点的出度为 1)。

3.png

  • 外向树:树中每个点有且仅有一条入边(或者说每个点的入度为 1)。

4.png

基于基环树有一项基本操作,寻找基环树上的

下文将深入讲解如何使用深度搜索算法在无向图中查找环结构。

2. 查找环

在图中查找的常用的算法有:

  • 深度搜索算法。
  • 广度搜索算法。
  • 并查集。
  • 拓扑排序算法。

广度深度搜索是原生算法,面对较复杂的图结构时,略显拙劣。较优的方案是使用优化过的并查集拓扑排序算法,其在性能和技巧上都很精妙。

Tips: 关于并查集和拓扑排序算法可查阅我的相关博文。

2.1 问题分析

起点和终点相同的路径称为回路或称为环(Cycle),除第一个顶点与最后一个顶点之外,其他顶点不重复出现的回路称为简单回路,本文仅对简单回路进行讨论。

Tips: 一般而言,回路至少需要 3 个顶点。

图中是否有环的前提条件:边数至少要等于顶点数。

所以对于有 n个顶点的无向(有向)图,是否存在环,可以先检查边的数量 m与顶点数量之间的关系。

Tips: 本文主要以无向图为讨论对象。

  1. 如果 m=n-1。可以是下面的几种情况。
  • 一个连通分量图情况。可以理解为以任何一个顶点为根节点构建成的树结构,此时连通分量为1,显然此情况无法成环。

    Tips: 所谓连通图,指任意两个顶点之间都有路径的图。

7.png

5.png

  • 上文说过,如果存在回路,顶点数量和边数至少要相等,这句话不能狭义地诠释为如果边数小于顶点数,则图中无环。

    m=n-1时,顶点数可以拆分成 n=m+1。这里遵循一个拆分原则,尽可能在已有边数的基础上成全图中某些顶点成环的要求。

    如下图所示,连通分量有 2个,其中一个子图中存在一个环。

    所以当边数少于顶点数时,需要讨论是否存在子图。

14.png

也可以是如下几种图形:

8.png

10.png

  1. m<n-1时,如果 m<=2则无法构建成环。其它情况下都能构建出一个有环的子图。

15.png

  1. m>=n 因为边数已经大于顶点数,显然图中有环。

11.png

12.png

13.png

2.2 深度搜索算法思想

2.2.1 连通分量

可以使用深度搜索算法查找图中是否有环,先了解一下深度搜索算法的搜索流程。

Tips: 深度搜索算法可以使用递归和非递归实现。本质是使用栈进行数据存储。

  • 先创建一个栈(非递归实现),用来存储搜索到的顶点。

16.png

  • A顶点为出发点,且把A顶点压入栈中,用于初始化栈。并在图的A顶点上做已入栈的标志。

17.png

  • 查询出与A顶点相邻的顶点B、E,并且压入栈中。

18.png

  • 继续查询与E顶点相邻的顶点(即入栈操作完成后,再查询与栈顶元素相邻的顶点)D,且压入栈中。

19.png

  • 检查与D顶点相邻的顶点C,且压入栈中。

20.png

经过上述操作后,可发现图结构中所有顶点全部被压入栈中,此时,能证明什么?

至少有一点是可以证明的:栈中的顶点在一个集合或在一个连通分量上。

使用如上搜索方案时,是否能找到此连通分量上的所有顶点?

先看下图的搜索结果:

当查询与 D顶点相邻的顶点时,因 DC没有连通性,故C无法入栈。栈中的顶点在一个连通分量上,是不容质疑的,而实际上 C也是此连通分量上的一份子。

21.png

所以,使用仅入栈不出栈的搜索方案,无法找到同一个连通分量上的所有顶点。

可以把深度搜索方案改一下,采用入栈、出栈形式。

  • 初始 A入栈。

29.png

  • A出栈,找到与A相邻的顶点B、E,且入栈。

30.png

  • E出栈,找到与E相邻的D顶点,且让其入栈。

31.png

  • D出栈,因没有与D相邻的顶点(每个顶点只能入一次栈)可入栈。继续B出栈,因C是与之相邻的顶点且没有入过栈,C入栈。

32.png

  • 最后C出栈,栈空。

33.png

至此,可以得到一个结论:

  • 在一次深度搜索过程中,入过栈的顶点都在一个集合中(或一个连通分量上)。
  • 使用出栈、入栈方案时,可以保证搜索到一个连通分量上的所有顶点。

**Tips: **使用广度搜索一样能找到一个连通分量上的所有顶点。

原理很简单,深度搜索是按纵深方式进行搜索的(类似于一条线上的蚂蚱),在互相有关联的纵深线上的顶点能被搜索到。

如下图所示:从A开始,有左、右两条纵深线,在搜索完左边的纵深线后。

25.png

可以继续搜索另一条纵深线。

26.png

一次深度搜索只能检查到一条连通分量。如果图中存在多个连通分量时,需要使用多次深度搜索算法。如下图所示:

23.png

连通分量与找环有什么关系?

连通分量与环之间有很一个简单的关系:一定是存在一个连通分量上。找环之前,先要确定目标顶点是不是在一个连通分量上。否则,都不在一起,还找什么环?

是否在一个连通分量上的顶点一定有环?

如下图,所有顶点都在一个连通分量中,实际情况是,图没有环。所以,仅凭顶点全部在一个连通分量上,是无法得到图中一定有环的结论。

33.png

那么,使用深度搜索算法在图中搜索时,怎么证明图中有环?

根据前面的推断,可以得出结论:

  • 所有搜索的顶点在一个连通分量上,且图或子图边数大于或等于顶点数时,那么图或子图中必然存在环。

23.png

那么?环上的顶点有什么特点?

如果图中存在环,那么,环上每个顶点必然至少有 2条边分别和环上另 2 个顶点相连,边的数量决定与其相邻顶点的数量。

Tips:道理很简单,如果有 n个人通过手牵手围成一个圈,如果其中有一个人只原意出一只手,则圈是无法形成的。 可以把此人移走,剩余人可以围成一个圈。

34.png

2.2.2 小结

查询环上的顶点时,需要满足如下几个要求:

  • 所有顶点在一个连通分量上。
  • 连通分量上的边数要大于或等于顶点数。
  • 环上至少有 2 条边的顶点可认定是环上的顶点。

2.3 编码实现

顶点类:

#include <iostream>
#include <vector>
#define MAXVERTEX  10
using namespace std;
/*
*顶点类
*/
struct Vertex {
	//编号
	int vid;
	//数据
	char val;
	//与之相邻的边数
	int edges;
	//是否访问过
	bool isVisited;
	//前驱节点编号 
	int pvid; 
    
	Vertex() {
		this->vid=-1;
		this->edges=0;
		this->isVisited=false;
	}

	Vertex(int vid,char val) {
		this->vid=vid;
		this->edges=0;
		this->val=val;
		this->isVisited=false;
	}
};

图类: 先在图类提供常规API(对顶点的维护函数,添加顶点和添加顶点之间的关系)

/*
* 图类
*/
class Graph {
	private:
		//所有顶点
		Vertex allVertex[MAXVERTEX];
		//二维矩阵,存储顶点关系(边)
		int matrix[MAXVERTEX][MAXVERTEX];
		//顶点数量
		int num;
	public:
		Graph() {
			this->num=0;
			for(int i=0; i<MAXVERTEX; i++) {
				for(int j=0; j<MAXVERTEX; j++) {
					this->matrix[i][j]=0;
				}
			}
		}
		/*
		*添加顶点,返回顶点编号
		*/
		int addVertex(char val) {
			Vertex ver(this->num,val);
			this->allVertex[this->num]=ver;
			return this->num++;
		}

		/*
		*添加顶点之间的关系
		*/
		void addEdge(int from,int to) {
			//无向图中,需要双向添加
			this->allVertex[from].edges++;
			this->matrix[from][to]=1;
			this->allVertex[to].edges++;
			this->matrix[to][from]=1;
		}

		/*
		*深度搜索算法找环
		*/
		void findCycle() { }
		/*
		*显示矩阵中边信息
		*/
		void showEdges() {
			for(int i=0; i<this->num; i++) {
				for(int j=0; j<this->num; j++) {
					cout<<this->matrix[i][j]<<"\t";
				}
				cout<<endl;
			}
		}
};

下图的子图A、B、E、D就是基环树。现使用代码构建下图:

34.png

//测试
int main() {
	Graph graph;//=new Graph();
	//添加顶点
	int aid= graph.addVertex('A');
	int bid= graph.addVertex('B');
	int cid= graph.addVertex('C');
	int did= graph.addVertex('D');
	int eid= graph.addVertex('E');
	//添加顶点之间关系
	graph.addEdge(aid,bid);  //A ---- B
	graph.addEdge(aid,eid);  //A  --- E
	graph.addEdge(bid,eid);  //B ---- E
	graph.addEdge(did,eid);  // E----D
	graph.showEdges();
	return 0;
}

确认图的构建是否正确。

35.png

核心函数: 使用深度搜索算法检查图中是否存在环。

/*
*深度搜索算法找环
*/
void findCycle(int from) {
    //记录连通分量中的边数
    int sides=0;
    //栈
    stack<int> myStack;
    //初始化栈
    myStack.push(from);
    //标志已经入过栈
    this->allVertex[from].isVisited=true;
    //存储搜索过的顶点
    vector<int> vers;

    //出栈操作
    while( !myStack.empty() ) {
        //出栈
        int vid= myStack.top();
        //存储搜索顶点
        vers.push_back(vid);
        //删除
        myStack.pop();
        //检查相邻节点
        for(int i=0; i<this->num; i++ ) {
            if( this->matrix[vid][i]==1 ) {
                // i 是 from 的相邻顶点
                sides++;
                //标志此边已经被记录
                this->matrix[vid][i]=-1;
                if(this->allVertex[i].isVisited==false ) {
                    //边对应顶点是否入过栈
                    myStack.push(i);
                    this->allVertex[i].isVisited=true;
                }
            }
        }
    }

    //输出搜索结果
    cout<<"输出连通分量中的顶点:"<<endl;
    for(int i=0; i<vers.size(); i++)
        cout<< this->allVertex[vers[i]].val<<"\t";
    cout<<endl;
    //存储搜索结果 
    allConns.push_back(vers);

    //检查环
    if(sides>=vers.size() && vers.size()>3 )  {
        //边数大于或等于搜索过的顶点数。表示搜索过的顶点在一个集合中,且有环
        cout<<"存在环:"<<endl;
        for(int i=0; i<vers.size(); i++) {
            if( this->allVertex[vers[i]].edges>=2 ) {
                cout<<this->allVertex[vers[i]].val<<"->\t";
            }
        }
        cout<<endl;
    }

    //检查是否还有其它连通分量
    for(int i=0; i<this->num; i++) {
        bool isExist=false;
        //是否已经搜索
        for(int j=0; j<allConns.size(); j++) {
            auto res = find( begin( allConns[j] ), end( allConns[j] ),this->allVertex[i].vid  );
            if(res!=end(allConns[j] )  ) {
                //搜索过
                isExist=true;
                break;
            }
        }
        if(!isExist) {
            findCycle(this->allVertex[i].vid );
        }
    }
}
//显示图中所有连通分量
void showConn() {
    cout<<"图中存在"<<allConns.size()<<"个连通分量"<<endl;
}

测试:

int main() {
    //省略……
	graph.showEdges();
	graph.findCycle(0);
	graph.showConn();
	graph.showEdges();
	return 0;
}

输出结果:

36.png

3. 总结

本文讲解怎么使用深度搜索算法在无向图中查找环,当然,也可以使用广度搜索算法实现。

检查图中连通性的最好的方案是使用并查集。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/468417.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

[Java 数据结构] 反射、枚举和lambda表达式

&#x1f389;&#x1f389;&#x1f389;点进来你就是我的人了 博主主页&#xff1a;&#x1f648;&#x1f648;&#x1f648;戳一戳,欢迎大佬指点!人生格言&#xff1a;当你的才华撑不起你的野心的时候,你就应该静下心来学习! 欢迎志同道合的朋友一起加油喔&#x1f9be;&am…

集成学习随机森林

集成学习 集成学习分为三类算法&#xff1a; 装袋法&#xff08;Bagging&#xff09;&#xff0c;提升法&#xff08;Boosting&#xff09;和融合stacking。 Bagging&#xff1a;核心思想是构建多个相互独立的评估器&#xff0c;然后对其预测进行平均或多数表决原则来决定集成…

极客星球|数据分析引擎黑马ClickHouse技术研究与实践

ClickHouse 在近几年是大数据分析引擎界的一匹黑马&#xff0c;从默默无闻到一路起飞&#xff0c;在 DB engine Rank 上进入前50名&#xff0c;成为全球数据引擎界耀眼的一颗明星。在全球范围内&#xff0c;ClickHouse 单表查询比其他引擎要快数倍以上&#xff0c;在过去的几年…

Vue+Echart实现利用率表盘效果【组件已封装,可直接使用】

效果演示 当利用超过70%&#xff08;可以自行设置&#xff09;&#xff0c;表盘变红 组件 里面对应两个图片资源&#xff0c;panelBackground_red.png 和 panelBackground_green.png&#xff0c;请前往百度网盘进行下载。如果喜欢其他颜色&#xff0c;可以使用.psd来修改导出…

redis从头开始【一】--面试的小伙伴必看

一 什么是NoSQL&#xff1f; Nosql not only sql&#xff08;不仅仅是SQL&#xff09; 关系型数据库&#xff1a;列行&#xff0c;同一个表下数据的结构是一样的。 非关系型数据库&#xff1a;数据存储没有固定的格式&#xff0c;并且可以进行横向扩展。 NoSQL泛指非关系型…

10.Yarn概述

如果说HDFS是存储&#xff0c;则Yarn就是cpu和内存&#xff0c;mapreduce就是程序。 1.基础架构 复习&#xff1a; 1.Container就是一个容器&#xff0c;其中封装了需要使用的内存与cpu 2.每当提交一个job,就会产生一个appMaster(总指挥),app Master负责其他container里面的…

【分布式技术专题】「分布式技术架构」手把手教你如何开发一个属于自己的Redis延时队列的功能组件

手把手教你如何开发一个属于自己的延时队列的功能组件 前提介绍解决痛点延时队列组件的架构延时队列组件的初始化流程延时队列组件的整体核心类架构延时队列组件的整体核心类功能 延时队列的开发组件延迟队列的机制配置初始化类源码 - DelayedQueueConfigurationRedission客户端…

排序算法 - 冒泡排序

文章目录 冒泡排序介绍冒泡排序实现复杂度和稳定性冒泡排序时间复杂度冒泡排序稳定性 代码实现核心&注意结尾 每日一道算法提高脑力&#xff0c;今天是第一天&#xff0c;来个最简单的算法–冒泡排序。 冒泡排序介绍 它是一种较简单的排序算法。它会遍历若干次要排序的数列…

jQuery知识点一

一、 jQuery 介绍 1.jQuery的概念&#xff1a; jQuery 是一个快速、简洁的 JavaScript 库&#xff0c;其设计的宗旨是“write Less&#xff0c;Do More”&#xff0c;即倡导写更少的代码&#xff0c;做更多的事情。 j 就是 JavaScript&#xff1b; Query 查询&#xff1b; 意思…

学习 Python 之 Pygame 开发魂斗罗(十五)

学习 Python 之 Pygame 开发魂斗罗&#xff08;十五&#xff09; 给魂斗罗游戏加入Boss1. 分析boss2. 创建boss类3. 在主类中加载Boss4. 修改子弹类逻辑&#xff0c;让boss可以开火5. 修改主类逻辑&#xff0c;让boss正常开火 给魂斗罗游戏加入Boss 在上次的博客学习 Python 之…

如何在不重装系统的情况下换固态硬盘?

随着固态硬盘的价格不断下降&#xff0c;越来越多的计算机用户希望用固态硬盘替换老旧的机械硬盘以获得更好的性能。 但是常规方法就避免不了重装系统&#xff0c;用户配置文件、系统设置、个人文件和已安装的程序又需要重新配置一遍。此外&#xff0c;还可能重新遇到很多问题…

城市“一网统管”平台—智慧平安小区的场景应用

随着城市建设进程的不断加快&#xff0c;关于城市的智能化治理需求也随之增多。在国家发布的“十四五”规划中&#xff0c;已经明确指出&#xff0c;推进新型城市建设&#xff0c;推行城市运行一网统管。作为推动城市治理体系和治理能力现代化的重要探索&#xff0c;“一网统管…

Linux/Unix常见IO模型

阻塞&#xff08;Blocking I/O&#xff09;、非阻塞&#xff08;Non-Blocking I/O&#xff09;、IO多路复用&#xff08;I/O Multiplexing&#xff09;、 信号驱动 I/O&#xff08;Signal Driven I/O&#xff09;&#xff08;不常用&#xff09;和异步&#xff08;Asynchronous…

智能家居“落地者”:三翼鸟用场景方案持续链接大众消费

互联网分析沙龙(techxue)原创 作者 &#xff5c; 锡海 编辑 &#xff5c; 七喜 从上海车展再到AWE2023展会&#xff0c;只要有大型活动的地方&#xff0c;都能看到人潮汹涌的景象&#xff0c;久违的烟火气又回来了。数据显示&#xff0c;社会消费已出现较为强劲反弹&#xff0…

长知识了,mongo的时间居然这个样子

1、前言 最近一直在使用mongo数据库&#xff0c;前面文章也介绍了一直在做数据过期的事情&#xff0c; mongo中的数据过期时间之前在程序中增加了一个字段 【Springboot系列】项目启动时怎么给mongo表加自动过期索引 之前看到时间字段没有时区的信息&#xff0c;没有关注&a…

微服务注册中心选型:Zookeeper、Eureka、Nacos、Consul和Etcd

注册中心基本概念 什么是注册中心&#xff1f; 注册中心主要有三种角色&#xff1a; 服务提供者&#xff08;RPC Server&#xff09;&#xff1a;在启动时&#xff0c;向 Registry 注册自身服务&#xff0c;并向 Registry 定期发送心跳汇报存活状态。 服务消费者&#xff08…

怎么知道网站服务器有没有被攻击?

​  一个网站服务器遭到攻击可能会给企业带来巨大的金融损失&#xff0c;因此&#xff0c;企业需要及时发现服务器是否被攻击。但是&#xff0c;企业如何知道自己的服务器是否被攻击呢?下面&#xff0c;我们来看一些服务器被攻击的警告信号。 1.网络延迟增加 在网络攻击中&a…

记一次运气非常好的渗透到服务器的经历

平平无奇的客服平台 这个客服平台是有RCE的&#xff0c;如果上传到的不是oss服务器&#xff0c;存储在本地服务器的话 在返回端口的url是存在st2 root权限&#xff0c;由于是客服后台服务器&#xff0c;没有啥有用价值的信息 直接替换私钥连服务器 继续翻找有用的信息 配置文件…

Django性能监视工具django-silk的使用

django-silk 是一个轻量级的 Django 应用性能监视工具&#xff0c;可帮助您了解 Django 应用的性能瓶颈、数据库查询等问题。它可以使用在django前后端分离的项目中&#xff0c;直接通过请求后台API接口即可对性能进行监视。以下是 django-silk 的使用步骤&#xff1a; 1.安装…

资本认可 | 开源网安成为中国未来独角兽企业,引领软件安全不断发展

4月11日&#xff0c;第七届万物生长大会中国未来独角兽大会盛大召开&#xff0c;本次大会中国投资发展促进会创投专委会联合微链共同发布了《2023中国未来独角兽TOP100榜单》&#xff0c;开源网安成功入选榜单。 《2023中国未来独角兽TOP100榜单》瞄准近两年融资较为活跃或融资…