数据结构 —— 图的表示

news2025/4/9 16:28:44

数据结构 —— 图的表示

  • 图的基本概念
  • 图的表示

今天我们来看看一个较为复杂的数据结构:

图的基本概念

在计算机科学中,“图”(Graph)是一种重要的非线性数据结构,用于表示对象集合及对象之间的多种关系。它由顶点(Vertex)和边(Edge)组成,能够有效地模拟和解决许多现实世界的问题,如网络路由、任务调度、社交网络分析等。下面是计算机中图概念的几个关键点:

  1. 定义:图 G=(V,E) 由两个主要部分构成:顶点集 V 和边集 E。顶点代表图中的实体或对象,边则表示顶点间的某种关系。顶点集 V(G) 和边集 E(G) 分别表示图 G 的顶点集合和边集合。
  1. 无向图与有向图
  • 无向图:图中的边没有方向,表示顶点间的对称关系。如果顶点 u 和顶点 v 之间有一条边,则从 uv 和从 vu 的关系是相同的。
  • 有向图:图中的边具有方向,表示从一个顶点到另一个顶点的单向关系。一条从顶点 u 指向顶点 v 的边表示一种特定方向的关系。
  1. 完全图:在一个有 n 个顶点的图中,如果每一对不同的顶点之间都恰好有一条边相连,则这个图被称为完全图。无向完全图有 n(n-1)/2 条边,有向完全图有 n(n-1) 条边。
  1. 带权图:图中的边可以被赋予一个数值,称为权重(Weight),这可以代表距离、成本、时间或其他任何度量。带权图在解决诸如最短路径问题时特别有用。
  1. 应用:图在计算机科学中有广泛的应用,包括但不限于:
  • 路径查找:如Dijkstra算法、A*算法用于寻找两点间的最短路径。
  • 网络流:如最大流/最小割问题,应用在网络设计、资源分配等领域。
  • 图的遍历:深度优先搜索(DFS)、广度优先搜索(BFS)用于访问图中所有顶点。
  • 图的连通性:检测图是否连通,计算连通分量等。
  • 社交网络分析:发现社群结构、影响力分析等。
  • 推荐系统:基于用户或物品间的关系构建推荐模型。

图作为数据结构的灵活性和表达能力使其成为理解和解决复杂问题的强大工具。

在这里插入图片描述

图的表示

我们了解了图的概念,现在有一个问题,我们应该如何表示图呢,我们一般有两种方式:
图的表示方法主要有两种:邻接矩阵和邻接表。

  1. 邻接矩阵
    邻接矩阵是一种使用二维数组来表示图中顶点之间关系的方法。对于一个有n个顶点的图,我们可以创建一个n×n的矩阵A,矩阵中的元素A[i][j]表示图中第i个顶点到第j个顶点是否存在边。如果存在边,则A[i][j]=1(对于无权图)或A[i][j]=权重值(对于有权图);如果不存在边,则A[i][j]=0。邻接矩阵的优点是直观且易于实现图的遍历,但空间消耗较大,特别是当图稀疏时(边的数量远小于顶点数量的平方)。
    在这里插入图片描述
namespace martix
{
    // 定义一个图类模板,参数为顶点类型V,边权重类型W,最大权重值MAX_W,以及是否是有向图Direction
    template<class V,class W,
            W MAX_W,bool Direction = false>
    class Graph
    {
    public:
        // 构造函数,初始化邻接矩阵,接受顶点数组和顶点数量
        Graph(const V* vertexs, int n)
        {
            // 为顶点容器预留空间以提高效率
            _vertexs.reserve(n);
            // 遍历顶点数组,将每个顶点添加到_vertexs,并记录其索引到_index映射中
            for(int i = 0; i < n; i++)
            {
                _vertexs.push_back(vertexs[i]);
                _index[vertexs[i]] = i;
            }

            // 初始化邻接矩阵,大小为n*n,初始值为MAX_W表示无连接
            _martix.resize(n);
            for(auto &row : _martix)
            {
                row.resize(n, MAX_W);
            }
        }

        // 获取顶点的索引
        int GetVertexIndex(const V& v)
        {
            // 查找顶点在_index中的映射,存在则返回索引,否则抛出异常
            auto ret = _index.find(v);
            if(ret != _index.end())
            {
                return ret->second;
            }
            else
            {
                throw std::invalid_argument("不存在的顶点");
                return -1; // 实际不会执行到此行,因为已抛出异常
            }
        }

        // 内部使用的加边函数,根据源顶点和目标顶点索引以及权重添加边
        void _AddEdge(int srci, int desi, const W& w)
        {
            _martix[srci][desi] = w; // 设置边的权重
            // 如果图是非定向的,则对称地添加边
            if(Direction == false)
            {
                _martix[desi][srci] = w;
            }
        }

        // 公开的加边接口,通过顶点名称和权重调用内部加边函数
        void AddEdge(const V& srci, const V& desi, const W& w)
        {
            int src = GetVertexIndex(srci); // 获取源顶点索引
            int des = GetVertexIndex(desi); // 获取目标顶点索引
            _AddEdge(src, des, w);          // 调用内部函数添加边
        }

        // 打印图的邻接矩阵表示
        void Print()
        {
            // 先打印顶点标签
            std::cout << "  ";
            for(size_t i = 0; i < _vertexs.size(); i++)
            {
                std::cout << _vertexs[i] << " ";
            }
            std::cout << std::endl;

            // 遍历并打印邻接矩阵,MAX_W的位置用#代替
            for(int i = 0; i < _vertexs.size(); i++)
            {
                std::cout << i << " ";
                for(int j = 0; j < _vertexs.size(); j++)
                {
                    if(_martix[i][j] != MAX_W)
                        std::cout << _martix[i][j] << " ";
                    else
                        std::cout << "# ";
                }
                std::cout << std::endl;
            }
        }

    private:
        // 存储图的顶点
        std::vector<V> _vertexs;
        // 顶点到其在_vector中的索引的映射
        std::map<V,int> _index;
        // 邻接矩阵表示的图
        std::vector<std::vector<W>> _martix;
    };

    // 测试函数,创建并操作一个有向图实例
    void TestGraph1()
    {
        // 使用字符作为顶点标识,整数作为边权重,INT_MAX表示无边,图是有向的
        Graph<char, int, INT_MAX, true> g("0123", 4);
        // 添加边及其权重
        g.AddEdge('0', '1', 1);
        g.AddEdge('0', '3', 4);
        g.AddEdge('1', '3', 2);
        g.AddEdge('1', '2', 9);
        g.AddEdge('2', '3', 8);
        g.AddEdge('2', '1', 5);
        g.AddEdge('2', '0', 3);
        g.AddEdge('3', '2', 6);
        // 打印图
        g.Print();
    }
}

在这里插入图片描述

  1. 邻接表
    邻接表是另一种常用的图表示法,特别适合于表示稀疏图。它由一个顶点列表和一系列的边列表(链表)组成。顶点列表存储图中所有的顶点,而每个顶点对应一个链表,这个链表包含了所有与该顶点直接相连的其他顶点(邻居)。链表中的每个节点通常包含两个信息:指向邻居顶点的指针和边的权重(对于有权图)。邻接表的优点是节省空间,特别是在图稀疏的情况下,但访问特定边可能不如邻接矩阵直接。在这里插入图片描述
namespace link_table
{
    // 定义边的结构体,包含目的顶点索引、权值和指向下一个边的指针
    template<class W>
    struct Edge
    {
        int _desIndex; // 目的顶点索引
        W _w;          // 边的权值
        Edge<W>* _next; // 指向下一个边的指针

        // 构造函数,初始化边的基本信息
        Edge(int desIndex, const W& w)
            : _desIndex(desIndex), _w(w), _next(nullptr) {}
    };

    // 图类模板,采用邻接表表示图
    template<class V,class W,bool Direction = false>
    class Graph
    {
    public:
        // 类型定义,简化Edge类型的引用
        typedef Edge<W> Edge;

        // 构造函数,初始化顶点信息和邻接表
        Graph(const V* vertexs, size_t n)
        {
            _vertexs.reserve(n); // 为顶点容器预留空间
            // 遍历顶点数组,添加顶点到_vertexs,并建立顶点到索引的映射
            for(size_t i = 0; i < n; i++)
            {
                _vertexs.push_back(vertexs[i]);
                _index[vertexs[i]] = i;
            }
            // 初始化邻接表,每个顶点对应的链表头指针初始化为空
            _martix.resize(n, nullptr);
        }

        // 获取顶点索引
        size_t GetVertexIndex(const V& v)
        {
            auto ret = _index.find(v);
            // 如果找到顶点,则返回其索引,否则抛出异常
            if(ret != _index.end())
            {
                return ret->second;
            }
            else
            {
                throw std::invalid_argument("不存在的顶点");
                return -1; // 实际不会执行到此行,因为已抛出异常
            }
        }

        // 添加边,根据顶点名称和权值在图中添加边
        void AddEdge(const V& src, const V& des, W w)
        {
            size_t srcIndex = GetVertexIndex(src);
            size_t desIndex = GetVertexIndex(des);

            // 创建新边,并将其插入源顶点的链表头部
            Edge* edge = new Edge(desIndex, w);
            edge->_next = _martix[srcIndex];
            _martix[srcIndex] = edge;

            // 若图无向,则对称地在目标顶点链表中也插入边
            if(Direction == false)
            {
                Edge* edgeRev = new Edge(srcIndex, w);
                edgeRev->_next = _martix[desIndex];
                _martix[desIndex] = edgeRev;
            }
        }

        // 打印图的邻接表表示
        void Print()
        {
            // 遍历每个顶点,打印其邻接链表
            for(size_t i = 0; i < _vertexs.size(); i++)
            {
                std::cout << _vertexs[i] << "[" << i << "]:";

                Edge* cur = _martix[i]; // 当前顶点的链表头指针
                while(cur)
                {
                    // 打印目的顶点索引和权值,以及指向下一个边的箭头
                    std::cout << _vertexs[cur->_desIndex] << "[" << cur->_desIndex << "]" 
                              << cur->_w << "->";
                    cur = cur->_next; // 移动到链表的下一个节点
                }
                std::cout << "nullptr" << std::endl; // 链表结束标记
            }
        }

    private:
        // 存储图的顶点
        std::vector<V> _vertexs;
        // 顶点到索引的映射
        std::map<V,int> _index;
        // 邻接表,每个元素是一个指向边的指针
        std::vector<Edge*> _martix;
    };

    // 测试函数,创建并操作一个无向图实例
    void TestGraph1()
    {
        std::string a[] = { "张三", "李四", "王五", "赵六" };
        // 创建图实例,并添加边
        Graph<std::string, int> g1(a, 4);
        g1.AddEdge("张三", "李四", 100);
        g1.AddEdge("张三", "王五", 200);
        g1.AddEdge("王五", "赵六", 30);
        // 打印图
        g1.Print();
    }
}

在这里插入图片描述

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

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

相关文章

【瑞吉外卖 | day01】项目介绍+后台登录退出功能

文章目录 瑞吉外卖 — day011. 所需知识2. 软件开发整体介绍2.1 软件开发流程2.2 角色分工2.3 软件环境 3. 瑞吉外卖项目介绍3.1 项目介绍3.2 产品原型展示3.3 技术选型3.4 功能架构3.5 角色 4. 开发环境搭建4.1 数据库环境搭建4.2 Maven项目构建 5. 后台系统登录功能5.1 创建需…

昇思25天学习打卡营第14天|GPT2文本摘要

一、简介&#xff1a; GPT-2&#xff08;Generative Pre-trained Transformer 2&#xff09;是由OpenAI开发的一种基于Transformer架构&#xff08;decoder-only&#xff09;的大型自然语言处理模型。它通过在大规模文本数据上进行预训练&#xff0c;能够理解和生成自然语言文…

Profibus DP主站转Modbus网关连接智能化电表通讯

Profibus DP主站转Modbus网关&#xff08;XD-MDPBM20&#xff09;&#xff0c;是实现不同工业通信协议之间互联互通的设备&#xff0c;主要将Profibus DP协议转换为Modbus协议&#xff0c;实现数据的双向传输。通过Profibus DP主站转Modbus网关&#xff08;XD-MDPBM20&#xff…

Java--创建对象内存分析

1.如图所示&#xff0c;左边为一个主程序&#xff0c;拥有main方法&#xff0c;右边定义了一个Pet类&#xff0c;通过debug不难看出&#xff0c;当启动该方法时&#xff0c;有以下该步骤 1.运行左边的实例化Pet类对象 2.跳转至右边定义Pet类的语句 3.跳回至左边获取Pet类基本属…

ch32v103xx 烧写

https://www.wch.cn/downloads/file/328.html?time2022-08-08%2004:49:39&codeqw2Y6RBMD5HJasIIBmtZu2t5GHi5BTDrnwjo4Ldi 最小可运行系统&#xff1b;但不建议按此使用&#xff0c;参官方说明 CH32V 基于 risv 架构&#xff0c;用arm swd不可以烧&#xff0c;ch32f 基…

机械拆装-基于Unity-装配功能的实现

目录 1. 装配场景的相机控制 2. 鼠标拖拽和旋转功能的实现 2.1 鼠标拖拽 2.2 物体旋转 3. 零件与装配位置的对应关系 4. 轴向装配的准备位置 5. 装配顺序的实现 5.1 标签提示 5.2 定义一个变量记录步骤数值 1. 装配场景的相机控制 开始装配功能时&#xff0c;需要将相机调…

WebStorm 2024 for Mac JavaScript前端开发工具

Mac分享吧 文章目录 效果一、下载软件二、开始安装1、双击运行软件&#xff08;适合自己的M芯片版或Intel芯片版&#xff09;&#xff0c;将其从左侧拖入右侧文件夹中&#xff0c;等待安装完毕2、应用程序显示软件图标&#xff0c;表示安装成功3、打开访达&#xff0c;点击【文…

加油卡APP开发,汽车加油省钱新模式

随着社会生活水平的提高&#xff0c;汽车已经成为了家家户户的出行工具&#xff0c;汽车加油也就成为了居民日常出行必不可少的开销。为了让居民享受到更加便利、优惠的加油体验&#xff0c;加油卡APP由此产生&#xff0c;不仅方便了用户&#xff0c;也给汽车加油市场提供了更加…

分子AI预测赛Task2笔记

下面所述比较官方的内容都来自官方文档 ‍‌⁠‌‍​​​‌​​⁠​​​​​&#xfeff;​​​&#xfeff;‍‬​​‍⁠‍‍​​‬​&#xfeff;‌​​​‌‍‬​​​​​​‍‌Task2&#xff1a;赛题深入解析 - 飞书云文档 (feishu.cn) 赛题背景 强调了人工智能在科研领域&…

算法训练营day24--93.复原IP地址 +78.子集 +90.子集II

一、93.复原IP地址 题目链接&#xff1a;https://leetcode.cn/problems/restore-ip-addresses/ 文章讲解&#xff1a;https://programmercarl.com/0093.%E5%A4%8D%E5%8E%9FIP%E5%9C%B0%E5%9D%80.html 视频讲解&#xff1a;https://www.bilibili.com/video/BV1fA4y1o715 1.1 初…

高温下的稳定选择 —— PP消解管,耐化学更耐用

PP消解管&#xff0c;即聚丙烯材质的消解管&#xff0c;是一种常用于化学分析中的实验室设备&#xff0c;主要用于样品的消解处理。以下是PP消解管的一些主要特性和应用&#xff1a; 主要特性&#xff1a; 1. 耐化学腐蚀&#xff1a;PP材料对多数酸、碱和有机溶剂具有良好的耐…

Keil5 ST-LINK setting闪退问题解决

1. 官网下载新版驱动文件 MDK uVision crashes when using ST-Link debugger 2. 解压替换 STLinkUSBDriver6.1.2.0Signed 我的库文件目录&#xff1a; D:\Tool\Keil5\ARM\STLink

Vue3快速上手--3小时掌握

1. Vue3简介 2020年9月18日&#xff0c;Vue.js发布版3.0版本&#xff0c;代号&#xff1a;One Piece&#xff08;n经历了&#xff1a;4800次提交、40个RFC、600次PR、300贡献者官方发版地址&#xff1a;Release v3.0.0 One Piece vuejs/core截止2023年10月&#xff0c;最新的…

数组-长度最小的子数组

M长度最小的子数组&#xff08;leetcode209&#xff09; /*** param {number} target* param {number[]} nums* return {number}*/ var minSubArrayLen function(target, nums) {const n nums.length;let ans n 1;let sum 0; // 子数组元素和let left 0; // 子数组…

燃料电池混合电源的能量管理系统

这个例子显示了燃料电池混合电源的能量管理系统。 这个例子展示了燃料电池混合电源的能量管理系统。 电路描述 本文给出了基于燃料电池的多电动飞机应急动力系统的仿真模型。随着MEA中起落架和飞控系统的电气化程度的提高&#xff0c;常规应急电源系统(冲压式空气涡轮或空气驱…

友好前端vue脚手架

企业级后台集成方案vue-element-admin-CSDN博客在哔站学习&#xff0c;老师说可以有直接的脚手架&#xff08;vue-element-admin&#xff09;立马去搜索&#xff0c;找到了这博主这篇文章 介绍 | vue-element-admin​​​​​​ 官方默认英文版&#xff1a; git clone https:/…

试用笔记之-Delphi xe 微信/支付宝支付源代码

首先delphi xe 微信/支付宝支付源代码下载&#xff1a; http://www.htsoft.com.cn/download/DelphiXEWeiXin_ZhiFuBao_ZhiFu.rar 解压后可以看到源代码 直接执行可执行文件&#xff1a;

消防认证-防火卷帘

一、消防认证 消防认证是指消防产品符合国家相关技术要求和标准&#xff0c;且通过了国家认证认可监督管理委员会审批&#xff0c;获得消防认证资质的认证机构颁发的证书&#xff0c;消防产品具有完好的防火功能&#xff0c;是住房和城乡建设领域验收的重要指标。 二、认证依据…

墨烯的C语言技术栈-C语言基础-003

三.数据类型 1.char // 字符数据型 2.short // 短整型 3.int // 整型 4.long // 长整型 5.long long // 更长的整型 6.float // 单精度浮点数 7.double // 双精度浮点数 为什么写代码? 为了解决生活中的问题 购物,点餐,看电影 为什么有这么多类型呢? 因为说的话都是字符型…

【详解】RV1106使用RKMPI+Yolov5部署检测

系列文章目录 第一篇&#xff1a;【详解】RV1106移植opencv-mobile库 文章目录 系列文章目录[TOC](文章目录) 前言一、烧入镜像二、项目工程1.获取源码编译2.移植项目文件 前言 记录使用RKMPI和Yolov5实现目标检测的demo。官方的资料比较详细&#xff1a;https://wiki.luckfo…