【数据结构周周练】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

相关文章

Spring IOC的三种实现方式详解

《SpringIOC的三种实现方式详解》:本文主要介绍SpringIOC的三种实现方式,在Spring框架中,IOC通过依赖注入来实现,而依赖注入主要有三种实现方式,构造器注入、Setter注入... 目录1. 构造器注入(Cons编程tructor Injection)2. Setter注入(Setter

Python实现文件下载、Cookie以及重定向的方法代码

《Python实现文件下载、Cookie以及重定向的方法代码》本文主要介绍了如何使用Python的requests模块进行网络请求操作,涵盖了从文件下载、Cookie处理到重定向与历史请求等多个方面,... 目录前言一、下载网络文件(一)基本步骤(二)分段下载大文件(三)常见问题二、requests模块处理

Java中注解与元数据示例详解

《Java中注解与元数据示例详解》Java注解和元数据是编程中重要的概念,用于描述程序元素的属性和用途,:本文主要介绍Java中注解与元数据的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参... 目录一、引言二、元数据的概念2.1 定义2.2 作用三、Java 注解的基础3.1 注解的定义3.2 内

vscode保存代码时自动eslint格式化图文教程

《vscode保存代码时自动eslint格式化图文教程》:本文主要介绍vscode保存代码时自动eslint格式化的相关资料,包括打开设置文件并复制特定内容,文中通过代码介绍的非常详细,需要的朋友... 目录1、点击设置2、选择远程--->点击右上角打开设置3、会弹出settings.json文件,将以下内

Python判断for循环最后一次的6种方法

《Python判断for循环最后一次的6种方法》在Python中,通常我们不会直接判断for循环是否正在执行最后一次迭代,因为Python的for循环是基于可迭代对象的,它不知道也不关心迭代的内部状态... 目录1.使用enuhttp://www.chinasem.cnmerate()和len()来判断for

JavaScript中的isTrusted属性及其应用场景详解

《JavaScript中的isTrusted属性及其应用场景详解》在现代Web开发中,JavaScript是构建交互式应用的核心语言,随着前端技术的不断发展,开发者需要处理越来越多的复杂场景,例如事件... 目录引言一、问题背景二、isTrusted 属性的来源与作用1. isTrusted 的定义2. 为

使用Python实现操作mongodb详解

《使用Python实现操作mongodb详解》这篇文章主要为大家详细介绍了使用Python实现操作mongodb的相关知识,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、示例二、常用指令三、遇到的问题一、示例from pymongo import MongoClientf

SQL Server使用SELECT INTO实现表备份的代码示例

《SQLServer使用SELECTINTO实现表备份的代码示例》在数据库管理过程中,有时我们需要对表进行备份,以防数据丢失或修改错误,在SQLServer中,可以使用SELECTINT... 在数据库管理过程中,有时我们需要对表进行备份,以防数据丢失或修改错误。在 SQL Server 中,可以使用 SE

一文详解Python中数据清洗与处理的常用方法

《一文详解Python中数据清洗与处理的常用方法》在数据处理与分析过程中,缺失值、重复值、异常值等问题是常见的挑战,本文总结了多种数据清洗与处理方法,文中的示例代码简洁易懂,有需要的小伙伴可以参考下... 目录缺失值处理重复值处理异常值处理数据类型转换文本清洗数据分组统计数据分箱数据标准化在数据处理与分析过

Redis多种内存淘汰策略及配置技巧分享

《Redis多种内存淘汰策略及配置技巧分享》本文介绍了Redis内存满时的淘汰机制,包括内存淘汰机制的概念,Redis提供的8种淘汰策略(如noeviction、volatile-lru等)及其适用场... 目录前言一、什么是 Redis 的内存淘汰机制?二、Redis 内存淘汰策略1. pythonnoe