随着学习的深入,我们的知识也在不断的扩展丰富。树结构有没有让大家蒙圈呢?相信我,学完图以后你就会觉得二叉树简直是简单得没法说了。其实我们说所的树,也是图的一种特殊形式。

图的概念

还记得我们学习树的第一篇文章时看到的那张关于树的图片吗?

在当时,我们就说过,图c 不是一颗树,而是一个图。为什么呢?从树的定义我们可以看出,树只能有一个根结点,平级之间不能有联系,可以有多个子结点。而图就不用遵守这些规则,图的特点就是结点之间都可以互相有联系。比如下图这样的都是图。

在上面所画的图中,图b 是的箭头的,而 图a 的连接线是没有箭头的,像这样有明确的方向的指向的图就叫做 有向图 。而没有箭头的,也就是没有方向指向的图就叫作 无向图 。

我们先将目光移到 图a-1 ,其实它就是把 图a 旋转了一下。大家能看出来了吗?如果忽略掉结点 4 和 1 之间的连线,那么它就是一颗树。是不是和我们上面关于树的图中的 图c 的概念一致了。

关于图的比较正式的官方定义是:

图(Graph)G 由两个集合 V 和 E 组成,记为 G=(V, E) ,其中 V 是顶点的有穷非空集合,E 是 V 中顶点的有穷集合,这些顶点偶对称为边。

在 有向图 中,连接两点的那个线段,从开始的结点到指向的那个结点可以记为 <x, y> 。<x, y> 和 <y, x> 是两个不同的边,也可以叫作 弧 。根据 图a ,我们可以看到这个图中有 <1, 2> 、 <2, 1> 、 <1, 3> 、 ️, 1> 、 <1, 4> 、 <4, 1> 、 ️, 4> 、 <4, 3> 这几条边。而 图b 中,因为它是有向图,所以它的边只有 <1, 2> 、 <1, 3> 、 ️, 4> 、 <4, 1> 这四条。

是不是感觉在看上面的图片的时候还比较清晰,一看这个定义就一脸懵逼了?像这种定义,如果你是需要考试的同学,那就还是要背下来的。如果只是像我一起想以学习应用或者了解为主的话,就不用去死记硬背了。V 就是结点,E 就是这些这些结点之间的关系,两个顶点之间的关系,也就是图上的那些连接结点的线段就是边。

OK,这三个最最基础的概念搞明白了,我们就继续学习其它的和图有关的那一大车术语!

图的相关术语

首先,我们用 n 来表示图中顶点的数目,用 e 来表示边的数目,记住这两个代号。

  • (1)子图:假设有两个图 G=(V, E) 和 G'=(V', E') 如果 V' 包含于 V 且 E' 包含于 E ,则称G' 为 G 的子图

上图中右边的那些子图都是属于原图的子图,可以看出子图可以产生非常多的形态,有向图 也是相同的概念,不过相对于 无向图 来说,有向图能够生成的子图更少一些,因为它的边是有方向的。

  • (2) 无向完全图和有向完全图:对于无向图,若具有 n(n-1)/2 条边,就是无向完全图。对于有向图,若具有 n(n-1) 条孤,就称为有向完全图。(参考完全二叉树)

其实完全图的概念就是图中所有相邻的结点都有边能够连结在一起。

对于 有向图 来说,虽说边是有方向的,当然我们也可以定义一个来回的方向,比如 <1, 2> 和 <2, 1> ,在有向图中我们就要画上两个相反方向的箭头表示可以从1到2也可以从2到1。而 无向图 中则是用一个边来代替这两个边的概念了,本身的那一条没有箭头方向的边就是双向的。

  • (3) 稀疏图和稠密图:有很少条边或孤(如e<nlog2n)的图称为稀疏图,反之称为稠密图

  • (4) 权和网:在实际应用中,每条边或孤可以标上具有某种意义的值,就像地图上的距离一样,这些数值就称为权。带权的图就可以称为网

最上方的的图片上 图a-2 和 图b-1 的边上的数字代表的就是权重。这两张图就可以称为网图。权重的概念我们后面在讲相关的算法时会学习到,从这两张图中,我们其实就可以很明显的看出,如果要从 结点1 走到 结点4 的话,并不是直接走 <1, 4> 这条边,而是走 <1, 3> 、 ️, 4> 这条路线更快些。

  • (5) 邻接点:两个有边的结点就是邻接点

  • (6) 度、入度和出度:顶点 v 的度就是指和 v 相关联的边的数目。对于有向图来说,箭头指向其它结点的就是出度,指向自己的就是入度

还是继续来看 图b 。结点1 有两个出度,一个入度。这个貌似不用解释太多了吧。

  • (7) 路径和路径长度:从某一个顶点到另一个顶点所经过的所有顶点就是路径。如果是有向图,那么它的路径就是按照箭头的方向。路径长度就是一条路径上经过的边或孤的数量

  • (8) 回路或环:第一个顶点和最后一个顶点相同的路径称为回路或环

  • (9) 连通、连通图和连通分量:如果某两个结点之间是有路径的,就称这两个结点是连通的。如果整个图中所有的结点都可以是互相连通的,则这个图就是连通图。连通分量就是无向图非连通图中的极大连通子图。

包括后面的三个概念也在这张图中一并给出了。在 无向图 中,连通分量就等于极大连通子图,在这个图中,我们有两个连通分量。

  • (10) 极大连通子图:连通子图所能含有的最大结点数,如果再增加一个结点那么这个子图就不是连通图了

  • (11) 生成树:一个极小连通子图,它含有图中全部的顶点,但只有足以构成一颗树的 n-1 条边,这样的连通子图就是连通图的生成树

其实就是通过一条路径,能够让图中所有的结点串联起来。在连通分量的图中,我们就根据两个连通分量生成了两个最小生成树。它们的 连通分量1 的生成树的结点并不一定非要是这种结构,我们可以让 结点4 在 结点2 下,这取决于我们如何遍历来生成这颗最小生成树。

最上面我们的 图a 的最小生成树其实就可以是 图a-1 去掉那条红色虚线。当然,也可以让 结点4 也在 结点1 下面,同样也是取决于我们的程序要如何遍历图来生成什么样的树。

  • (12)生成森林:在非连通图中,每一个连通分量都可以生成一个连通生成树,这样就构成了整个非连通图的生成森林

是不是看完之后晕头转向了?没关系,这些术语我们在后面的学习中将会经常用到,而且这还不是最全面的。大家可以根据参考书目和其它学习资料来对图的相关术语进行更加深入的学习和理解。

总结

图的概念介绍得差不多了,大家可以消化消化再继续学习后面的内容。这只是个开始,不少同学会不会觉得这玩意对比 树 结构一下子又提升了好多。不用怕,在学习完后面的知识后,即使你暂时还没有搞明白 图 相关的内容,但你一定对 树 结构的理解会更加深入了。为什么呢?树 其实就是没有回路的图,它们的遍历无外乎都是通过深度或者广度来实现的,只是图更复杂一点而已。这下是不是感觉未来还是有点希望的啦?学习,往往是一个渐进的过程,当前的知识和过去的知识总会有所关联的,先不用想太多,一步一步的踏实走下去吧!

参考资料:

《数据结构》第二版,严蔚敏

《数据结构》第二版,陈越

《数据结构高分笔记》2020版,天勤考研

关注公众号:【硬核项目经理】获取最新文章

添加微信/QQ好友:【xiaoyuezigonggong/149844827】免费得PHP、项目管理学习资料

知乎、公众号、抖音、头条搜索【硬核项目经理】

B站ID:482780532

【PHP数据结构】图的概念和存储结构的更多相关文章

  1. Java数据结构——树的三种存储结构

    (转自http://blog.csdn.net/x1247600186/article/details/24670775) 说到存储结构,我们就会想到常用的两种存储方式:顺序存储和链式存储两种. 先来 ...

  2. 【C语言--数据结构】线性表链式存储结构

    直接贴代码 头文件 #ifndef __LINKLIST_H__ #define __LINKLIST_H__ typedef void LinkList; typedef struct _tag_L ...

  3. 【PHP数据结构】图的存储结构

    图的概念介绍得差不多了,大家可以消化消化再继续学习后面的内容.如果没有什么问题的话,我们就继续学习接下来的内容.当然,这还不是最麻烦的地方,因为今天我们只是介绍图的存储结构而已. 图的顺序存储结构:邻 ...

  4. 图的存储结构与操作--C语言实现

    图(graph)是一种比树结构还要复杂的数据结构,它的术语,存储方式,遍历方式,用途都比较广,所以如果想要一次性完成所有的代码,那代码会非常长.所以,我将分两次来完成图的代码.这一次,我会完成图的五种 ...

  5. C/C++中float和double的存储结构(转)

    在C/C++中float是32位的,double是64位的,两者在内存中的存储方式和能够表示的精度均不同,目前C/C++编译器标准都遵照IEEE制定的浮点数表示法来进行float,double运算. ...

  6. C#与数据结构--图的遍历

    http://www.cnblogs.com/abatei/archive/2008/06/06/1215114.html 8.2 图的存储结构 图的存储结构除了要存储图中各个顶点的本身的信息外,同时 ...

  7. Oracle 存储结构

    数据库是存储数据的容器,它的主要功能是保存和共享数据. oracle数据库的存储结构可以分为逻辑存储结构和物理存储结构,对于这两种存储结构,oracle是分别进行管理的. 逻辑存储结构:oracle内 ...

  8. 图的存储结构大赏------数据结构C语言(图)

    图的存储结构大赏------数据结构C语言(图) 本次所讲的是常有的四种结构: 邻接矩阵 邻接表 十字链表 邻接多重表 邻接矩阵 概念 两个数组,一个表示顶点的信息,一个用来表示关联的关系. 如果是无 ...

  9. 算法与数据结构(四) 图的物理存储结构与深搜、广搜(Swift版)

    开门见山,本篇博客就介绍图相关的东西.图其实就是树结构的升级版.上篇博客我们聊了树的一种,在后边的博客中我们还会介绍其他类型的树,比如红黑树,B树等等,以及这些树结构的应用.本篇博客我们就讲图的存储结 ...

随机推荐

  1. Easylogging++的使用及扩展

    目录 简介 使用 扩展 配置日志路径 时间滚动日志 自动删除日志 封装到一个头文件 源代码优化(不推荐) 附件 简介 Easylogging++ 是用于 C++ 应用程序的单头高效日志库.它非常强大, ...

  2. NOIP 模拟 $27\; \rm 牛半仙的妹子图$

    题解 \(by\;zj\varphi\) 颜色数很少,考虑枚举颜色数. 建出来一棵最小生成树,可以证明在最小生成树上,一个点到另一个点的路径上的最大权值最小(易证,考虑 \(\rm kruskal\) ...

  3. docker 安装部署 redis(配置文件启动)

    获取 redis 镜像 docker pull redis:4.0.12 docker images 创建容器 创建宿主机 redis 容器的数据和配置文件目录 # 创建宿主机 redis 容器的数据 ...

  4. js 遍历数组对象求和

    这个通常是求多个商品的总价遇到的情形: [ 0: {id: 1, name: "服务费", price: "1.00"} 1: {id: 2, name: &q ...

  5. 线程池ExecutorService的使用

    转载自: 海子 Java并发编程:线程池的使用 在前面的文章中,我们使用线程的时候就去创建一个线程,这样实现起来非常简便,但是就会有一个问题: 如果并发的线程数量很多,并且每个线程都是执行一个时间很短 ...

  6. servlet中servletContext的五大作用(五)

    1.    获取web的上下文路径 2.    获取全局的参数 3.    作为域对象使用 4.    请求转发 5.    读取web项目的资源文件 package day10.about_serv ...

  7. vue@cli3 public目录下的静态图片,如何使用在css类文件中(sass可行,纯css不行)

    之前写了一篇vue文件怎么使用的文章,有人问我怎么在css文件中使用public下的文件,这是个好问题,因为我之前都没有研究过 需要解决的2个问题 一开始按照vue文件的使用方式(https://ww ...

  8. Go测试--子测试

    目录 简介 简单的例子 子测试命名规则 过滤筛选 子测试并发 总结 简介 简单的说,子测试提供一种在一个测试函数中执行多个测试的能力,比如原来有TestA.TestB和TestC三个测试函数,每个测试 ...

  9. String转double失去精度问题

    最近遇到一个坑,微信小程序中退款 19.9的字符串转double变成19.89,导致退不成功 . 坑死我了.现在把更改后的代码贴出来 public static void main(String[] ...

  10. Ubuntu 设置不更新某些软件

    方法来自:https://blog.csdn.net/zhrq95/article/details/79527073 保持某软件版本不变,如我wps-office,(已测有效@Ubuntu 16.04 ...