【数据结构周周练】029 判断无向图是否为一棵树算法原理详解及代码分享

本文主要是介绍【数据结构周周练】029 判断无向图是否为一棵树算法原理详解及代码分享,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、题目

设计一个算法,判断一个图G是否为一棵树,如果是,返回TRUE,否则,返回FALSE。

二、美丽的星座

星座真的好美好美。特别是当人类给它们赋予含义的那一刻,更美,仿佛有了灵魂一般。

 是不是很美,是不是?你以为我是让你过来看星星的吗?你以为我是希望你以后能够好好学习天文学知识的吗?当然不仅仅是这样啦!细心的你应该知道星星多么像我们学的图啊!!!

三、分析

敲黑板!!!

看题了嗨,学习完了我们可以一起畅游星空哈,现在回来我们的重心,看下面这两个星座,为了方便观看,研究其特性,我们放小一点,它们不仅仅是一个星座,同样它们也是一个图,同样它们也是一棵树,也就是说这两个图都是一棵树,那它们有什么特点呢?

该无向图是一棵树

 第一个图有 7颗星星,相当于有三个顶点,星星之间有  6条连线,相当于有  6条边;

第二个图有11颗星星,相当于有11个顶点,星星之间有10条连线,相当于有10条边。

这是树的特性,有n个顶点,就会有n-1条边,且每个顶点都有边相连。即一个图是一棵树的条件是:有n个顶点,就会有n-1条边且每个顶点都有边相连

所以我们的算法就在于判断当我们遍历完图之后,我们是否访问完所有的结点,并且,访问的结点的条数是结点个数-1。

当然,如果我们采用邻接表存储的图,那每一条边都会访问两次,也就是说,我们访问边的次数是边数的两倍。

第一步,遍历图之前的操作

我们遍历图,因为我们要做判断,判断结点是否被访问,被访问说明图中有环,就不是树,所以我们需要定义一个数组来保存结点访问情况,并设置所有值为false,当访问结点后,该结点对应的数组位置的元素改为true。

for (int i = 0; i < G.vexnum; i++) //将所有的访问状态设置为false,即未访问。visited[i] = false;

并且我们要判断访问的边以及结点数量,如果结点访问的数量同图中结点数量一致,并且访问过的边的次数是顶点个数减一的两倍,说明是一棵树,所以我们还需要设置两个变量,用来存储访问结点次数以及访问边的次数。

int VNum = 0;//访问的顶点的数量
int ENum = 0;//访问的边的数量

第二步,遍历图

遍历图我们采用递归算法,为了方便,我们单独写一个遍历算法DFS,即深度优先遍历图(Depth-First-Search)。遍历过程中,每遍历一次,将一个结点的访问改为true,顶点数目+1。

visited[v] = true;//做访问标记
VNum++; //顶点计数+1

 然后获取当前结点,即v结点的第一个邻接点,如果存在,说明有一条边存在,即需要边的数量+1,然后访问v结点的第一个邻接点,如果该邻接点未访问,则继续递归访问,如果访问过了,则访问下一个邻接点。

int w = FirstNeighbor(G, v);
while (w!=-1)
{ENum++;if (!visited[w])DFS(G, v, VNum, ENum, visited);w = NextNeighbor(G, v, w);
}

第三步,做判断

如果满足遍历的定点数与图的顶点数相同并且访问的边数和等于顶点数-1的两倍,则说明是树,否则不是树。

	if (VNum == G.vexnum&& ENum == 2 * (G.vexnum - 1))return true;elsereturn false;

四、全部代码

刚才上面是分析,为了便于大家理解,我将分析对应的代码也写上去了,为了方便大家整体分析,接下来我将所有的代码分享一下。

bool GraphIsTree(Graph &G) {for (int i = 0; i < G.vexnum; i++){visited[i] = false;}int VNum = 0;int ENum = 0;if (VNum == G.vexnum&& ENum == 2 * (G.vexnum - 1))return true;elsereturn false;
}void DFS(Graph &G, int v, int &VNum, int &ENum, int visited[]) {visited[v] = true;VNum++;int w = FirstNeighbor(G, v);while (w!=-1){ENum++;if (!visited[w])DFS(G, v, VNum, ENum, visited);w = NextNeighbor(G, v, w);}
}

大家有什么问题在下面留言哦!!!

这篇关于【数据结构周周练】029 判断无向图是否为一棵树算法原理详解及代码分享的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/295401

相关文章

JDK9到JDK21中值得掌握的29个实用特性分享

《JDK9到JDK21中值得掌握的29个实用特性分享》Java的演进节奏从JDK9开始显著加快,每半年一个新版本的发布节奏为Java带来了大量的新特性,本文整理了29个JDK9到JDK21中值得掌握的... 目录JDK 9 模块化与API增强1. 集合工厂方法:一行代码创建不可变集合2. 私有接口方法:接口

Spring 缓存在项目中的使用详解

《Spring缓存在项目中的使用详解》Spring缓存机制,Cache接口为缓存的组件规范定义,包扩缓存的各种操作(添加缓存、删除缓存、修改缓存等),本文给大家介绍Spring缓存在项目中的使用... 目录1.Spring 缓存机制介绍2.Spring 缓存用到的概念Ⅰ.两个接口Ⅱ.三个注解(方法层次)Ⅲ.

Spring Boot 整合 Redis 实现数据缓存案例详解

《SpringBoot整合Redis实现数据缓存案例详解》Springboot缓存,默认使用的是ConcurrentMap的方式来实现的,然而我们在项目中并不会这么使用,本文介绍SpringB... 目录1.添加 Maven 依赖2.配置Redis属性3.创建 redisCacheManager4.使用Sp

Spring Cache注解@Cacheable的九个属性详解

《SpringCache注解@Cacheable的九个属性详解》在@Cacheable注解的使用中,共有9个属性供我们来使用,这9个属性分别是:value、cacheNames、key、key... 目录1.value/cacheNames 属性2.key属性3.keyGeneratjavascriptor

PyTorch中cdist和sum函数使用示例详解

《PyTorch中cdist和sum函数使用示例详解》torch.cdist是PyTorch中用于计算**两个张量之间的成对距离(pairwisedistance)**的函数,常用于点云处理、图神经网... 目录基本语法输出示例1. 简单的 2D 欧几里得距离2. 批量形式(3D Tensor)3. 使用不

Python模拟串口通信的示例详解

《Python模拟串口通信的示例详解》pySerial是Python中用于操作串口的第三方模块,它支持Windows、Linux、OSX、BSD等多个平台,下面我们就来看看Python如何使用pySe... 目录1.win 下载虚www.chinasem.cn拟串口2、确定串口号3、配置串口4、串口通信示例5

电脑系统Hosts文件原理和应用分享

《电脑系统Hosts文件原理和应用分享》Hosts是一个没有扩展名的系统文件,当用户在浏览器中输入一个需要登录的网址时,系统会首先自动从Hosts文件中寻找对应的IP地址,一旦找到,系统会立即打开对应... Hosts是一个没有扩展名的系统文件,可以用记事本等工具打开,其作用就是将一些常用的网址域名与其对应

Nginx 413修改上传文件大小限制的方法详解

《Nginx413修改上传文件大小限制的方法详解》在使用Nginx作为Web服务器时,有时会遇到客户端尝试上传大文件时返回​​413RequestEntityTooLarge​​... 目录1. 理解 ​​413 Request Entity Too Large​​ 错误2. 修改 Nginx 配置2.1

springboot项目redis缓存异常实战案例详解(提供解决方案)

《springboot项目redis缓存异常实战案例详解(提供解决方案)》redis基本上是高并发场景上会用到的一个高性能的key-value数据库,属于nosql类型,一般用作于缓存,一般是结合数据... 目录缓存异常实践案例缓存穿透问题缓存击穿问题(其中也解决了穿透问题)完整代码缓存异常实践案例Red

Dubbo之SPI机制的实现原理和优势分析

《Dubbo之SPI机制的实现原理和优势分析》:本文主要介绍Dubbo之SPI机制的实现原理和优势,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Dubbo中SPI机制的实现原理和优势JDK 中的 SPI 机制解析Dubbo 中的 SPI 机制解析总结Dubbo中