news 2026/4/23 17:40:29

图的基础概念操作与遍历

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
图的基础概念操作与遍历

一、图的基础概念与术语

  1. 概念:图是一种非线性数据结构,由顶点和边组成,相较于线性关系(链表)和分治关系(树),网络关系(图)的自由度更高,因而更为复杂。

  2. 常见类型

    1. 是否有方向:无向图与有向图

      • 在无向图中,边表示两顶点之间的“双向”连接关系
      • 在有向图中,边具有方向性,即A→B和B→A两个方向的边是相互独立的
    2. 所有顶点是否连通:连通图和非连通图

      • 对于连通图,从某个顶点出发,可以到达其余任意顶点。
      • 对于非连通图,从某个顶点出发,至少有一个顶点无法到达。
    3. 我们还可以为边添加“权重”变量,从而得到有权图

  3. 术语:

    1. 邻接:当两顶点之间存在边相连时,称这两顶点“邻接”。
    2. 路径:从顶点 A 到顶点 B 经过的边构成的序列被称为从 A 到 B 的“路径”。
    3. 度:一个顶点拥有的边数。对于有向图,入度表示有多少条边指向该顶点,出度表示有多少条边从该顶点指出。

二、图的表示

  1. 邻接矩阵:设图的顶点数量为 ,邻接矩阵使用一个 大小的矩阵来表示图,每一行(列)代表一个顶点,矩阵元素代表边,用0或1表示两个顶点之间是否存在边。

  1. 邻接表:邻接表(adjacency list)使用n个链表来表示图,链表节点表示顶点。第i个链表对应顶点i,其中存储了该顶点的所有邻接顶点(与该顶点相连的顶点)。

三、图的基础操作

  1. 基于邻接矩阵的实现
/* 基于邻接矩阵实现的无向图类 */ class GraphAdjMat { vector<int> vertices; vector<vector<int>> adjMat; public: /* 构造方法 */ GraphAdjMat(const vector<int> &vertices, const vector<vector<int>> &edges) { // 添加顶点 for (int val : vertices) { addVertex(val); } // 添加边 for (const vector<int> &edge : edges) { addEdge(edge[0], edge[1]); } } /* 获取顶点数量 */ int size() const { return vertices.size(); } /* 添加顶点 */ void addVertex(int val) { int n = size(); vertices.push_back(val); adjMat.emplace_back(vector<int>(n, 0)); for (vector<int> &row : adjMat) { row.push_back(0); } } /* 删除顶点 */ void removeVertex(int index) { if (index >= size()) { throw out_of_range("顶点不存在"); } vertices.erase(vertices.begin() + index); adjMat.erase(adjMat.begin() + index); for (vector<int> &row : adjMat) { row.erase(row.begin() + index); } } /* 添加边 */ // 参数 i, j 对应 vertices 元素索引 void addEdge(int i, int j) { if (i < 0 || j < 0 || i >= size() || j >= size() || i == j) { throw out_of_range("顶点不存在"); } adjMat[i][j] = 1; adjMat[j][i] = 1; } /* 删除边 */ void removeEdge(int i, int j) { if (i < 0 || j < 0 || i >= size() || j >= size() || i == j) { throw out_of_range("顶点不存在"); } adjMat[i][j] = 0; adjMat[j][i] = 0; } /* 打印邻接矩阵 */ void print() { cout << "顶点列表 = "; printVector(vertices); cout << "邻接矩阵 =" << endl; printVectorMatrix(adjMat); } };
  1. 基于邻接表的实现
/* 基于邻接表实现的无向图类 */ class GraphAdjList { public: unordered_map<Vertex *, vector<Vertex *>> adjList; /* 在 vector 中删除指定节点 */ void remove(vector<Vertex *> &vec, Vertex *vet) { for (int i = 0; i < vec.size(); i++) { if (vec[i] == vet) { vec.erase(vec.begin() + i); break; } } } /* 构造方法 */ GraphAdjList(const vector<vector<Vertex *>> &edges) { for (const vector<Vertex *> &edge : edges) { addVertex(edge[0]); addVertex(edge[1]); addEdge(edge[0], edge[1]); } } /* 获取顶点数量 */ int size() { return adjList.size(); } /* 添加边 */ void addEdge(Vertex *vet1, Vertex *vet2) { if (!adjList.count(vet1) || !adjList.count(vet2) || vet1 == vet2) throw invalid_argument("不存在顶点"); adjList[vet1].push_back(vet2); adjList[vet2].push_back(vet1); } /* 删除边 */ void removeEdge(Vertex *vet1, Vertex *vet2) { if (!adjList.count(vet1) || !adjList.count(vet2) || vet1 == vet2) throw invalid_argument("不存在顶点"); remove(adjList[vet1], vet2); remove(adjList[vet2], vet1); } /* 添加顶点 */ void addVertex(Vertex *vet) { if (adjList.count(vet)) return; adjList[vet] = vector<Vertex *>(); } /* 删除顶点 */ void removeVertex(Vertex *vet) { if (!adjList.count(vet)) throw invalid_argument("不存在顶点"); adjList.erase(vet); for (auto &adj : adjList) { remove(adj.second, vet); } } /* 打印邻接表 */ void print() { cout << "邻接表 =" << endl; for (auto &adj : adjList) { const auto &key = adj.first; const auto &vec = adj.second; cout << key->val << ": "; printVector(vetsToVals(vec)); } } };

四、图的遍历

  1. 广度优先搜索:广度优先遍历是一种由近及远的遍历方式,从某个节点出发,始终优先访问距离最近的顶点,并一层层向外扩张
  2. 代码实现:
vector<Vertex *> graphBFS(GraphAdjList &graph, Vertex *startVet) { vector<Vertex *> res; unordered_set<Vertex *> visited = {startVet}; queue<Vertex *> que; que.push(startVet); while (!que.empty()) { Vertex *vet = que.front(); que.pop(); res.push_back(vet); for (auto adjVet : graph.adjList[vet]) { if (visited.count(adjVet)) continue; que.push(adjVet); visited.emplace(adjVet); } } return res; }
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/23 11:19:05

wangEditor支持跨平台ppt图片批量转存操作

680元打造企业级Word一键粘贴CMS系统 - .NET程序员实战指南 各位老铁&#xff0c;我是河北一名"头发日渐稀疏"的.NET程序员&#xff0c;最近接了个CMS官网项目&#xff0c;客户要加Word一键粘贴功能。预算680元&#xff1f;没问题&#xff01;看我怎么用"技术抠…

作者头像 李华
网站建设 2026/4/23 11:27:22

JS如何利用分块技术实现超大附件的上传优化?

北京XX软件公司涉密项目大文件传输解决方案&#xff08;基于SM4国密算法的多数据库兼容方案&#xff09; 一、项目背景与核心需求深化 作为服务政府及军工领域的软件企业&#xff0c;我司当前涉密项目需满足以下严苛要求&#xff1a; 多数据库兼容&#xff1a;需无缝适配达梦…

作者头像 李华
网站建设 2026/4/23 11:26:39

网页页面如何设计JSP大文件上传的进度条?

大文件传输解决方案&#xff08;源码级交付&#xff09; 作为山西IT行业软件公司项目负责人&#xff0c;我深刻理解当前需求的复杂性与紧迫性。针对政府、央企客户对100G级文件传输、高稳定性断点续传、信创兼容、数据安全的核心诉求&#xff0c;结合集团多项目统一组件、低成…

作者头像 李华
网站建设 2026/4/23 11:27:11

四旋翼无人机PID控制仿真模型探索

四旋翼无人机PID控制仿真模型 模型&#xff1a;四旋翼无人机动力学模型。 包含力方程组与力矩方程组 控制策略&#xff1a;用经典PID控制算法对其内环姿态和外环位置进行控制 内环姿态环&#xff0c;外环位置环 报告&#xff1a;有建模和仿真报告&#xff0c;很详细&#xff0c…

作者头像 李华
网站建设 2026/4/23 11:28:10

使用MediaCreationTool修复/升级到Windows10系统

1、点此下载MediaCreationTool 。官网地址https://go.microsoft.com/fwlink/?LinkId691209百度网盘百度网盘: https://pan.baidu.com/s/1vYRZB6vUvUNUK-bCGZN6Tw?pwd9999提取码: 99992、下载完成后&#xff0c;鼠标双击运行MediaCreationTool.exe。3、允许此应用对你的设备进…

作者头像 李华
网站建设 2026/4/22 11:43:10

55、Ubuntu 系统软件管理全攻略

Ubuntu 系统软件管理全攻略 在 Ubuntu 系统中,软件管理是一项重要的日常操作。下面我们将详细介绍如何通过不同工具和命令来管理软件,包括安装、更新、搜索以及从源代码编译软件等方面。 Synaptic 软件包管理器的使用 在屏幕左侧类别下方,有四个按钮,分别是“Sections”…

作者头像 李华