【数据结构与算法——TypeScript】图结构(Graph)

【数据结构与算法——TypeScript】

图结构(Graph)

认识图结构以及特性

什么是图?

  • 在计算机程序设计中,图结构 也是一种非常常见的数据结构。
    • 但是,图论其实是一个非常大的话题
  • 认识一下关于图的一些内容
    • 图的抽象数据类型
    • 一些算法实现。
  • 什么是图?
    • 图结构是一种与树结构有些相似的数据结构。
    • 图论数学的一个分支,并且,在数学的概念上,树是图的一种。
    • 它以图为研究对象,研究 顶点 组成的图形的数学理论和方法
    • 主要研究的目的是事物之间的关系顶点代表事物代表两个事物间的关系
  • 我们知道树可以用来模拟很多现实的数据结构
    • 比如: 家谱/公司组织架构等等
  • 那么图长什么样子?
  • 或者什么样的数据使用图来模拟更合适呢?

图的现实案例

  • 人与人之间的关系网
    • 甚至科学家们在观察人与人之间的关系网时,还发现了六度空间理论
  • 六度空间理论
    • 理论上认为世界上任何两个互相不认识的两人。
    • 只需要很少的中间人就可以建立起联系。
    • 并非一定要经过6步,只是需要很少的步骤。
  • 地铁图

在这里插入图片描述

  • 村庄之间的关系网

在这里插入图片描述

再次 什么是图?

  • ❤️‍🔥 那么,什么是图呢?
    • 我们会发现,上面的节点(其实图中叫顶点Vertex)之间的关系,是不能使用树来表示
    • 使用任何的树结构都不可以模拟。
    • 这个时候,我们就可以使用图来模拟它们。
  • ❤️‍🔥 图通常有什么特点呢?
    • 💚 一组顶点:通常用 V (Vertex) 表示顶点的集合
    • 💚 一组边:通常用 E (Edge) 表示边的集合
      • ✓ 边是顶点和顶点之间的连线
      • ✓ 边可以是有向的,也可以是无向的。
      • ✓ 比如A — B,通常表示无向。 A --> B,通常表示有向

欧拉和七拉问题解法

历史故事

  • 8世纪著名古典数学问题之一。
    • 在哥尼斯堡的一个公园里,有七座桥普雷格尔河中两个岛岛与河岸连接起来(如图)。

      在这里插入图片描述

    • 有人提出问题: 一个人怎样才能不重复、不遗漏地一次走完七座桥,最后回到出发点。

  • 1735年,有几名大学生写信给当时正在俄罗斯的彼得斯堡科学院任职的瑞典天才数学家欧拉,请他帮忙解决这一问题。
    • 欧拉在亲自观察了哥伦斯堡的七桥后,认真思考走法,但是始终没有成功,于是他怀疑七桥问题是不是无解的。
    • 1736年29岁的欧拉向 彼得斯堡 科学院递交了《哥尼斯堡的七座桥》的论文,在解答问题的同时,开创了数学的一个新的分支——图论与几何拓扑,也由此展开了数学史上的新历程。

欧拉解答

  • 他不仅解决了该问题,并且给出了 连通图 可以一笔画的充要条件是:
    • 奇点的数目不是0个就是2 个
    • 连到一点的边的数目如果是奇数条,就称为奇点
    • 如果是偶数条就称为偶点
    • 要想一笔画成,必须中间点均是偶点
    • 也就是有来路必有另一条去路奇点只可能在两端,因此任何图能一笔画成,奇点要么没有,要么在两端
  • 个人思考:
    • 欧拉在思考这个问题的时候,并不是针对某一个特性的问题去考虑,而是将岛和桥抽象成了点和线
    • 抽象是数学的本质,而编程我们也一再强调抽象的重要性。
    • 汇编语言是对机器语言的抽象,高级语言是对汇编语言的抽象。
    • 操作系统是对硬件的抽象,应用程序在操作系统的基础上构建。

在这里插入图片描述

图结构的常见术语

关于术语的概述

  • 我们在学习树的时候,树有很多的相关术语

  • 了解这些术语有助于我们更好的理解树结构

  • 我们也来学习一下图相关的术语

    • 但是图的术语其实非常多,如果你找一本专门讲图的各个方面的书籍,会发现 只是术语 就可以 占据满满的一个章节。
    • 这里,我们先介绍几个比较常见的术语。
  • 我们先来看一个抽象出来的图
     用数字更容易我们从整体来观察整个图结构
    在这里插入图片描述

术语

  1. 顶点:

    • 顶点刚才我们已经介绍过了,表示图中的一个节点
    • 比如地铁站中某个站/多个村庄中的某个村庄/互联网中的某台主机/人际关系中的人
  2. 边:

    • 边刚才我们也介绍过了,表示顶点和顶点之间的连线
    • 比如地铁站中两个站点之间的直接连线,就是一个边。
    • ❗️ 注意: 这里的边不要叫做路径,路径有其他的概念,待会儿我们会介绍到。
    • 之前的图中: 0 - 1有一条边,1 - 2有一条边,0 - 2没有边。
  3. 相邻顶点:

    • 由一条边连接在一起的顶点称为相邻顶点
    • 比如0 - 1是相邻的,0 - 3是相邻的。 0 - 2是不相邻的。
  4. 度:

    • 一个顶点的度是相邻顶点的数量。
    • 比如0顶点和其他两个顶点相连,0顶点的度是2
    • 比如1顶点和其他四个顶点相连,1顶点的度是4
  5. 路径:

    • 路径是顶点v1,v2…,vn的一个连续序列,比如上图中0 1 5 9就是一条路径。
    • 简单路径: 简单路径要求不包含重复的顶点。 比如 0 1 5 9 是一条简单路径。
    • 回路: 第一个顶点和最后一个顶点相同的路径称为回路。 比如 0 1 5 6 3 0
  6. 无向图:

    • 上面的图就是一张无向图,因为所有的边都没有方向
    • 比如 0 - 1之间有变,那么说明这条边可以保证 0 -> 1,也可以保证 1 -> 0。
  7. 有向图:

    • 有向图表示的图中的边是有方向的。
    • 比如 0 -> 1,不能保证一定可以 1 -> 0,要根据方
      向来定。

在这里插入图片描述

  1. 无权图:

    • 我们上面的图就是一张无权图(边没有携带权重)
    • 我们上面的图中的边是没有任何意义
    • 不能说 0 - 1的边,比4 - 9的边更远或者用的时间更长。
  2. 带权图:

    • 带权图表示边有一定的权重
    • 这里的权重可以是任意你希望表示的数据
      ✓ 比如距离或者花费的时间或者票价

图的表示

  • 怎么在程序中表示图呢?
    • 我们知道一个图包含很多顶点,另外包含顶点和顶点之间的连线(边)
    • 这两个都是非常重要的图信息,因此都需要在程序中体现出来
  • 顶点的表示相对简单,我们先讨论顶点的表示。
    • 上面的顶点,我们抽象成了1 2 3 4,也可以抽象成A B C D。
    • 在后面的案例中,我们使用A B C D。
    • 那么这些A B C D我们可以使用一个数组来存储起来(存储所有的顶点)
    • 当然,A,B,C,D也可以表示其他含义的数据(比如村庄的名字).
  • 那么边怎么表示呢?
    • 因为边是两个顶点之间的关系,所以表示起来会稍微麻烦一些。
    • 下面,我们具体讨论一下变常见的表示方式。

邻接矩阵和邻接表

邻接矩阵

  • 一种比较常见的表示图的方式: 邻接矩阵
    • 邻接矩阵让每个节点和一个整数项关联,该整数作为数组的下标值
    • 我们用一个二维数组来表示顶点之间的连接
    • 二维数组 [0][2] -> A -> C
  • 画图演示:

在这里插入图片描述

  • 图片解析:
    • 在二维数组中,0表示没有连线,1表示有连线。
    • 通过二维数组,我们可以很快的找到一个顶点和哪些顶点有连线。(比如A顶点,只需要遍历第一行即可)
    • 另外,A - A,B - B(也就是顶点到自己的连线),通常使用0表示。
  • 邻接矩阵的问题:
    • 邻接矩阵还有一个比较严重的问题,就是如果图是一个稀疏图
    • 那么矩阵中将存在大量的0,这意味着我们浪费了计算机存储空间来表示根本不存在的边

邻接表

  • 另外一种常用的表示图的方式: 邻接表

    • 邻接表由图中每个顶点以及和顶点相邻的顶点列表组成。
    • 这个列表有很多种方式来存储: **数组/链表/字典(哈希表)**都可以。
  • 画图演示
    在这里插入图片描述

  • 图片解析:

    • 其实图片比较容易理解。
    • 比如我们要表示和A顶点有关联的顶点(边),A和B/C/D有边,
    • 那么我们可以通过A找到对应的数组/链表/字典,再取出其中的内容就可以啦。
  • 邻接表的问题:

    • 邻接表计算"出度"是比较简单的(出度: 指向别人的数量,入度: 指向自己的数量)
    • 邻接表如果需要计算有向图的"入度",那么是一件非常麻烦的事情。
    • 它必须构造一个**“逆邻接表”,才能有效的计算“入度”。但是开发中“入度”**相对用的比较少。

创建图类

  • 先创建Graph类
  • 👩🏻‍💻 代码解析
    • 创建Graph的构造函数,这个我们在封装其他数据结构的时候已经非常熟悉了。
  • 定义了两个属性:
    ✓ vertexes: 用于存储所有的顶点,我们说过使用一个数组来保存。
    ✓ adjList: adj是adjoin的缩写,邻接的意思。 adjList用于存储所有的边,我们这里采用邻接表的形式。
  • 之后,我们来定义一些方法以及实现一些算法就是一个完整的图类了。
    class Graph<T> {// 顶点vertexes: T[] = [];// 邻接表adjList: Map<T, T[]> = new Map();}

添加方法

  • 现在我们来增加一些添加方法。
    • 添加顶点: 可以向图中添加一些顶点。
    • 添加边: 可以指定顶点和顶点之间的边。
  • 添加顶点
    • 👩🏻‍💻 代码解析:
      • 我们将添加的顶点放入到数组中。
      • 另外,我们给该顶点创建一个数组[],该数组用于存储顶点连接的所有的边.(回顾邻接表的实现方式)
  • 添加边
    • 👩🏻‍💻 代码解析:
      • 添加边需要传入两个顶点,因为边是两个顶点之间的边,边不可能单独存在。
      • 根据顶点v取出对应的数组,将w加入到它的数组中。
      • 根据顶点w取出对应的数组,将v加入到它的数组中。
      • 因为我们这里实现的是无向图,所以边是可以双向的。
  // 添加顶点addVertex(vertex: T) {//   将顶点添加到顶点数组中保存this.vertexes.push(vertex);//   创建一个邻接表的数组this.adjList.set(vertex, []);}addEdge(v1: T, v2: T) {this.adjList.get(v1)?.push(v2);this.adjList.get(v2)?.push(v1);}

printEdges方法

  • 为了能够正确的显示图的结果,我们来实现一下Graph的printEdges方法

  • 测试代码:

      const graph = new Graph();graph.addVertex('A');graph.addVertex('B');graph.addVertex('C');graph.addVertex('D');graph.addVertex('E');graph.addVertex('F');graph.addVertex('G');graph.addVertex('H');graph.addVertex('I');graph.addEdge('A', 'B');graph.addEdge('A', 'C');graph.addEdge('A', 'D');graph.addEdge('C', 'D');graph.addEdge('C', 'G');graph.addEdge('D', 'G');graph.addEdge('D', 'H');graph.addEdge('B', 'E');graph.addEdge('B', 'F');graph.addEdge('E', 'I');graph.printEdges();
    • 运行结果:
      [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-fK63qbI7-1692005552724)(/Volumes/web/数据结构与算法(ts)/截图/截屏2023-08-14 15.37.06.png “运行结果”)]
      在这里插入图片描述

图的遍历

图的遍历思想

图的遍历思想和树的遍历思想是一样的。
图的遍历意味着需要将图中每个顶点访问一遍,并且不能有重复的访问

  • 有两种算法可以对图进行遍历
    • 广度优先搜索(Breadth-First Search,简称BFS)
    • 深度优先搜索(Depth-First Search,简称DFS)
  • 两种遍历算法,都需要明确指定第一个被访问的顶点
    • 它们的遍历过程分别是怎么样呢?
    • 我们以一个迷宫中关灯为例。
    • 现在需要你进入迷宫,将迷宫中的灯一个个关掉,你会怎么关呢?

遍历的思想

  • 两种算法的思想:
    • BFS: 基于队列,入队列的顶点先被探索。
    • DFS: 基于栈或使用递归,通过将顶点存入栈中,顶点是沿着路径被探索的,存在新的相邻顶点就去访问。
  • 为了记录顶点是否被访问过,我们使用三种颜色来反应它们的状态
    • 白色: 表示该顶点还没有被访问。
    • 灰色: 表示该顶点被访问过,但并未被探索过。
    • 黑色: 表示该顶点被访问过且被完全探索过。
  • 或者我们也可以使用Set来存储被访问过的节点。

广度优先搜索

  • 广度优先搜索算法的思路 💡:

    • 广度优先算法会从指定的第一个顶点开始遍历图,先访问其所有的相邻点,就像一次访问图的一层
    • 换句话说,就是先宽后深的访问顶点
  • 图解BFS
    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-mNFb1uIZ-1692005552726)(/Volumes/web/数据结构与算法(ts)/画图/图/广度优先搜索.png “广度优先搜索”)]
    在这里插入图片描述

    ◼ 广度优先搜索的实现:
    ◼ 创建一个队列Q。

  • 代码 💻 :

  // 广度优先bfs() {// 1. 判断是否有顶点if (this.vertexes.length === 0) return;// 2. 创建队列结构来访问每个顶点const queue: T[] = [];queue.push(this.vertexes[0]);// 3. 创建Set,来记录一个顶点是否被访问过const visited = new Set();visited.add(this.vertexes[0]);// 4. 遍历队列中每一个顶点while (queue.length) {//  4.1 访问队列中第一个顶点const vertex = queue.shift()!;console.log(vertex);//  4.2 取出相邻的顶点const neighbors = this.adjList.get(vertex);if (!neighbors) continue;for (const neighbor of neighbors) {if (!visited.has(neighbor)) {visited.add(neighbor);queue.push(neighbor);}}}}
  • 运行结果:

在这里插入图片描述

深度优先搜索

  • 深度优先搜索的思路:

    • 深度优先搜索算法将会从第一个指定的顶点开始遍历图,沿着路径知道这条路径最后被访问了。
    • 接着原路回退并探索下一条路径。
  • 深度优先搜索

    • 深度优先搜索算法的实现:
      • 广度优先搜索算法我们使用的是队列,这里可以使用完成,也可以使用递归。
  • 图DFS

    在这里插入图片描述

  • 代码 💻 :

  // 深度优先dfs() {// 1. 判断有没有顶点,没有直接返回if (this.vertexes.length === 0) return;// 2.创建栈结构const stack: T[] = [];stack.push(this.vertexes[0]);// 3. 创建Set结构const visited = new Set();visited.add(this.vertexes[0]);// 4. 从第一个顶点开始访问while (stack.length) {const vertex = stack.pop()!;console.log(vertex);// 取出相邻的顶点const neighbors = this.adjList.get(vertex);if (!neighbors) continue;for (let i = neighbors.length - 1; i >= 0; i--) {const nei = neighbors[i];if (!visited.has(nei)) {visited.add(nei);stack.push(nei);}}}}
  • 运行结果:

在这里插入图片描述

图结构的常见建模

  • 对交通流量建模
    • 顶点可以表示街道的十字路口,边可以表示街道。
    • 加权的边可以表示限速或者车道的数量或者街道的距离。
    • 建模人员可以用这个系统来判定最佳路线以及最可能堵车的街道。
  • 对飞机航线建模
    • 航空公司可以用图来为其飞行系统建模。
    • 将每个机场看成顶点,将经过两个顶点的每条航线看作一条边。
    • 加权的边可以表示从一个机场到另一个机场的航班成本,或两个机场间的距离。
    • 建模人员可以利用这个系统有效的判断从一个城市到另一个城市的最小航行成本。

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

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

相关文章

jmeter获取mysql数据

JDBC Connection Configuration Database URL: jdbc:mysql:// 数据库地址 /库名 JDBC Driver class&#xff1a;com.mysql.jdbc.Driver Username&#xff1a;账号 Password&#xff1a;密码 JDBC Request 字段含义 字段含义 Variable Name Bound to Pool 数据库连接池配置…

使用vue3 + ts + vite + v-md-editor 在前端页面预览markdown文件

1.效果预览 2. 依赖包安装 yarn add kangc/v-md-editornext v-md-editor中文官网&#xff1a;https://code-farmer-i.github.io/vue-markdown-editor/zh/ v-md-editor分为4种组件&#xff1a; 轻量版编辑器进阶版编辑器预览组件html预览组件 对UI组件库页面&#xff0c;我只需…

问道管理:缩量小幅上涨说明什么?

股市里面&#xff0c;股票价格上涨或跌落都是常见现象。可是关于那些在商场上寻求收益的出资者来说&#xff0c;他们需要对每一个股市中的价格动摇有深化的了解&#xff0c;以便做出更正确的出资决策。最近&#xff0c;出资者们发现商场缩量小幅上涨的现象时有发生&#xff0c;…

Jmeter压测实战:Jmeter二次开发之自定义函数

目录 1 前言 2 开发准备 3 自定义函数核心实现 3.1 新建项目 3.2 继承实现AbstractFunction类 3.3 最终项目结构 4 Jmeter加载扩展包 4.1 maven构建配置 4.2 项目打包 4.3 Jmeter加载扩展包 5 自定义函数调用调试 5.1 打开Jmeter函数助手&#xff0c;选择自定义函数…

clickhouse 删除操作

OLAP 数据库设计的宗旨在于分析适合一次插入多次查询的业务场景&#xff0c;市面上成熟的 AP 数据库在更新和删除操作上支持的均不是很好&#xff0c;当然 clickhouse 也不例外。但是不友好不代表不支持&#xff0c;本文主要介绍在 clickhouse 中如何实现数据的删除&#xff0c…

单链表相关操作(插入,删除,查找)

通过上一节我们知道顺序表的优点&#xff1a; 可随机存储&#xff08;O(1)&#xff09;&#xff1a;查找速度快 存储密度高&#xff1a;每个结点只存放数据元素&#xff0c;而单链表除了存放数据元素之外&#xff0c;还需存储指向下一个节点的指针 http://t.csdn.cn/p7OQf …

【2023年11月第四版教材】《第4章-信息系统管理(合集篇)》

第4章-信息系统管理之管理方法&#xff08;第四版新增章节&#xff09;&#xff08;第一部分&#xff09; 章节说明1 管理方法1.1 信息系统四个要素1.2 信息系统四大领域1.3 信息系统战略三角1.4 信息系统架构转换1.5 信息系统体系架构1.6 信息系统运行1.7 运行和监控1.8 管理和…

kafka基本概念及操作

kafka介绍 Kafka是最初由Linkedin公司开发&#xff0c;是一个分布式、支持分区的&#xff08;partition&#xff09;、多副本的 &#xff08;replica&#xff09;&#xff0c;基于zookeeper协调的分布式消息系统&#xff0c;它的最大的特性就是可以实时的处理大量数据以满足各…

【LeetCode】242 . 有效的字母异位词

242 . 有效的字母异位词&#xff08;简单&#xff09; 方法&#xff1a;哈希表 思路 首先判断两个字符串长度是否相等&#xff0c;不相等直接返回 false&#xff1b;接下来设置一个长度为26 的哈希表&#xff0c;分别对应26个小写字母&#xff1b;遍历两个字符串&#xff0c;…

Go语言工程实践之测试与Gin项目实践

Go 语言并发编程 及 进阶与依赖管理_软工菜鸡的博客-CSDN博客 03 测试 回归测试一般是QA(质量保证)同学手动通过终端回归一些固定的主流程场景 集成测试是对系统功能维度做测试验证,通过服务暴露的某个接口,进行自动化测试 而单元测试开发阶段&#xff0c;开发者对单独的函数…

KAFKA第二课之生产者(面试重点)

生产者学习 1.1 生产者消息发送流程 在消息发送的过程中&#xff0c;涉及到了两个线程——main线程和Sender线程。在main线程中创建了一个双端队列RecordAccumulator。main线程将消息发送给RecordAccumulator&#xff0c;Sender线程不断从RecordAccumulator中拉取消息发送到K…

03-基础入门-搭建安全拓展

基础入门-搭建安全拓展 1、涉及的知识点2、常见的问题3、web权限的设置4、演示案例-环境搭建&#xff08;1&#xff09;PHPinfo&#xff08;2&#xff09;wordpress&#xff08;3&#xff09;win7虚拟机上使用iis搭建网站&#xff08;4&#xff09;Windows Server 2003配置WEB站…

C#应用处理传入参数 - 开源研究系列文章

今天介绍关于C#的程序传入参数的处理例子。 程序的传入参数应用比较普遍&#xff0c;特别是一个随操作系统启动的程序&#xff0c;需要设置程序启动的时候不显示主窗体&#xff0c;而是在后台运行&#xff0c;于是就有了传入参数问题&#xff0c;比如传入/h或者/min等等。所以此…

YOLO v8目标跟踪详细解读(二)

上一篇&#xff0c;结合代码&#xff0c;我们详细的介绍了YOLOV8目标跟踪的Pipeline。大家应该对跟踪的流程有了大致的了解&#xff0c;下面我们将对跟踪中出现的卡尔曼滤波进行解读。 1.卡尔曼滤波器介绍 卡尔曼滤波&#xff08;kalman Filtering&#xff09;是一种利用线性…

欧拉OS 使用 CentOS 7 yum repo

一、下载CentOS的repo的yum文件 任何基于CentOS的yum的repo 的url是这样的&#xff1a; 但欧拉OS输出这个变量为&#xff1a;openEuler 20.03 (LTS-SP3) 那明显欧拉想要使用这个yum的url找不到这个版本&#xff0c; 所以直接讲这个变量替换为 7, Centos 7的7 然后执行&…

从零实战SLAM-第七课(多视角几何)

在七月算法报的班&#xff0c;老师讲的蛮好。好记性不如烂笔头&#xff0c;关键内容还是记录一下吧&#xff0c;课程入口&#xff0c;感兴趣的同学可以学习一下。 --------------------------------------------------------------------------------------------------------…

设计模式--策略模式

目录 一.场景 1.1场景 2.2 何时使用 2.3个人理解 二. 业务场景练习 2.1业务: 2.2具体实现 2.3思路 三.总结 3.1策略模式的特点&#xff1a; 3.2策略模式优点 3.3策略模式缺点 一.场景 1.1场景 许多相关的类仅仅是行为有异&#xff0c;也就是说业务代码需要根据场景不…

归并排序 与 计数排序

目录 1.归并排序 1.1 递归实现归并排序&#xff1a; 1.2 非递归实现归并排序 1.3 归并排序的特性总结: 1.4 外部排序 2.计数排序 2.1 操作步骤: 2.2 计数排序的特性总结: 3. 7种常见比较排序比较 1.归并排序 基本思想: 归并排序(MERGE-SORT)是建立在归并操作上的一种…

redis分布式集群-redis+keepalived+ haproxy

redis分布式集群架构&#xff08;RedisKeepalivedHaproxy&#xff09;至少需要3台服务器、6个节点&#xff0c;一台服务器2个节点。 redis分布式集群架构中的每台服务器都使用六个端口来实现多路复用&#xff0c;最终实现主从热备、负载均衡、秒级切换的目标。 redis分布式集…

使用Edge和chrom扩展工具(GoFullPage)实现整页面截图或生成PDF文件

插件GoFullPage下载&#xff1a;点击免费下载 如果在浏览网页时&#xff0c;有需要整个页面截图或导出PDF文件的需求&#xff0c;这里分享一个Edge浏览器的扩展插件&#xff1a;GoFullPage。 这个工具可以一键实现页面从上到下滚动并截取。 一、打开“管理扩展”&#xff08;…