【数据结构】图的存储(十字链表)

article/2025/8/4 21:13:53

弧节点 

  • tailvex数据域:存储弧尾一端顶点在顺序表中的位置下标;
  • headvex 数据域:存储弧头一端顶点在顺序表中的位置下标;
  • hlink 指针域:指向下一个以当前顶点作为弧头的弧;
  • tlink 指针域:指向下一个以当前顶点作为弧尾的弧;
  • info 指针:存储弧的其它信息,例如有向网中弧的权值。如果不需要存储其它信息,可以省略。

 顶点节点

  • data 数据域:用来存储顶点的信息;
  • firstin 指针域:指向一个链表,链表中记录的都是以当前顶点为弧头的弧的信息;
  • firstout 指针域:指向另一个链表,链表中记录的是以当前顶点为弧尾的弧的信息。

       就类似这种结构图:

大家别看他有这么多变量,其实理解起来很简单,十字链表是有向图的一种存储方式。

顶点节点中的firstout管理的是所有的弧尾,而弧节点的tlink负责链接。

顶点节点中的firstin管理的是所有的弧头 ,而弧节点的hlink负责链接。


在编写代码中,我给弧节点增加了两个指针分别是hprelink与tprelink。

这两个指针的意义是:

hprelink: 指向上一个以当前顶点作为弧头的弧;

tprelink:指向上一个以当前顶点作为弧尾的弧;

其实就是双链表的思想,为什么想用双链表,其实我在实现删除的时候:当我以头结点进行firstout遍历后找到当前顶点为弧尾的弧后,发现还要遍历一遍firstin找到当前顶点为弧头的弧。所以就寻思用双链表吧。其实如果不用双链表也大差不差只不过要多遍历一次,都是要找到这个节点的前一个节点和后一个节点,进行节点维护。

我觉得写代码中有一些心得值得分享的:在删除节点的时候吧,我总是把弧头与弧尾放到一块分析,比如:弧头前一个节点如果为空弧尾前一个节点如果不为空,弧头后一个节点如果为空弧尾后一个节点如果为空.......分析了一大坨。漏洞百出,逻辑没有闭合(痛苦了很长时间)。后来,我就逐个分析把他们分开了,删除这个节点的本质其实就是维护这个节点的前弧头一个节点,和后弧头一个节点,如果前为空....如果后为空....。之后分析弧尾也是这个套路。后来我总结一下:维护一个位置,其实就是分析前一个位置与后一个位置,不用管这个节点位置本身在头还是尾还是中间。

#pragma once
#include <string>
#include <vector>
#include <iostream>
#include <map>
using namespace std;namespace Cross_linked_list
{template<class W>struct ArcNode{int tailvex;    // 弧尾下标int headvex;    // 弧头下标ArcNode<W>* hlink; // 相同弧头的下一条弧ArcNode<W>* tlink;    // 相同弧尾的下一条弧ArcNode<W>* hprelink;  // 弧头的上一条弧ArcNode<W>* tprelink;    // 弧尾的上一个弧W weight;        // 权值ArcNode(int tail, int head, W w):tailvex(tail), headvex(head), hlink(nullptr), tlink(nullptr), hprelink(nullptr), tprelink(nullptr), weight(w){}};// 顶点结构节点template<class V, class W>struct VexNode{V data;      // 顶点信息ArcNode<W>* firstin; // 指向该顶点的第一条入弧ArcNode<W>* firstout;// 指向该顶点的第一条出弧VexNode(V val):data(val), firstin(nullptr), firstout(nullptr){}};template<class V, class W> class Graph{typedef ArcNode<W> Arc;  typedef VexNode<V, W> Vex; public:// 4 , "ABCD"Graph(int n, const V* ver){// 存储顶点并与下标建立映射_vertexs.reserve(n);_vexstable.resize(n, nullptr); // 初始化为nullptrfor (int i = 0; i < n; i++){_vertexs.push_back(ver[i]);_indexMap[ver[i]] = i;// 创建VexNode对象_vexstable[i] = new Vex(ver[i]); }}~Graph(){// 释放所有顶点和弧节点for (auto vex : _vexstable){// 释放出弧节点Arc* p = vex->firstout;while (p){Arc* temp = p;p = p->tlink;delete temp;}// 释放顶点delete vex;}}void addArc(V src, V dst, W weight){int srci = locateVex(src);int dsti = locateVex(dst);if (srci == -1 || dsti == -1){cout << "顶点不存在!" << endl;return;}Arc* arc = new Arc(srci, dsti, weight);// 获取源顶点和目标顶点的指针Vex* srcVex = _vexstable[srci]; Vex* dstVex = _vexstable[dsti]; // 维护出弧链表(双向)if (srcVex->firstout) {srcVex->firstout->tprelink = arc;}arc->tlink = srcVex->firstout;srcVex->firstout = arc;arc->tprelink = nullptr;// 维护入弧链表(双向)if (dstVex->firstin) {dstVex->firstin->hprelink = arc;}arc->hlink = dstVex->firstin;dstVex->firstin = arc;arc->hprelink = nullptr;}void delEdge(V src, V dst){int srci = locateVex(src);int dsti = locateVex(dst);if (srci == -1 || dsti == -1){cout << "顶点不存在!" << endl;return;}// 查找从srci到dsti的节点Arc* arc = _vexstable[srci]->firstout;Arc* prev = nullptr;while (arc && arc->headvex != dsti){prev = arc;arc = arc->tlink;}if (!arc){cout << "边不存在!" << endl;return;}// 获取源顶点和目标顶点的指针Vex* srcVex = _vexstable[srci];Vex* dstVex = _vexstable[dsti];// 从出弧链表中删除// 如果前节点存在,不存在就更新头节点if (prev)prev->tlink = arc->tlink;elsesrcVex->firstout = arc->tlink;// 如果后节点存在if (arc->tlink)arc->tlink->tprelink = prev;// 从入弧链表中删除if (arc->hprelink)arc->hprelink->hlink = arc->hlink;elsedstVex->firstin = arc->hlink;// 如果后节点存在if (arc->hlink)arc->hlink->hprelink = arc->hprelink;delete arc;}void print(){cout << "映射关系" << endl;for (size_t i = 0; i < _vexstable.size(); ++i){cout << _vexstable[i]->data << ":" << i << endl;}cout << endl;cout << "出弧链表:" << endl;for (size_t i = 0; i < _vexstable.size(); ++i){Arc* p = _vexstable[i]->firstout;cout << "顶点 " << _vexstable[i]->data << "(" << i << ") 的出弧:";while (p){cout << "[" << _vertexs[p->headvex] << ", 权值:" << p->weight << "] ";p = p->tlink;}cout << endl;}cout << endl;cout << "入弧链表:" << endl;for (size_t i = 0; i < _vexstable.size(); ++i){Arc* p = _vexstable[i]->firstin;cout << "顶点 " << _vexstable[i]->data << "(" << i << ") 的入弧:";while (p){cout << "[" << _vertexs[p->tailvex] << "->" << _vexstable[i]->data << ", 权值:" << p->weight << "] ";p = p->hlink;}cout << endl;}}private:// 查找顶点在顶点表中的下标int locateVex(V v){auto it = _indexMap.find(v);if (it != _indexMap.end()){return it->second;}else{return -1;}}private:vector<Vex*> _vexstable;  // 顶点表map<V, int> _indexMap;    // 映射关系vector<V> _vertexs;       // 顶点集合};void test(){Graph<char, int> g(4, "ABCD"); g.addArc('A', 'B', 1);g.addArc('A', 'C', 1);g.addArc('C', 'D', 1);g.addArc('C', 'A', 1);g.addArc('D', 'A', 1);g.addArc('D', 'C', 1);cout << "\n删除边之前:\n";g.print();g.delEdge('C', 'A');g.delEdge('C', 'D');cout << "\n删除边之后:\n";g.print();}
}

效果展示: 


http://www.hkcw.cn/article/UGNflAljBT.shtml

相关文章

(二)微服务(grpc/grpc消费者)

文章目录 项目地址一、grpc介绍1.1 项目初始化1. 创建grpc项目2. 项目结构二、Discount grpc创建2.1 实体层1. Coupon实体2.2 Protos1. 创建discount.proto2. 配置proto3. 创建DiscountService4. Program里注册服务2.3 Seed 数据1. 创建表和Seed数据2. 自动migration2.4 更新Do…

机电的焊接技术

焊接技术:高温或高压条件下,使用焊接材料(焊条或焊丝)将两块或两块以上的母材(待焊接的工件)连接 成一个整体的操作方法&#xff61; 2.3.1 焊接设备和焊接材料的分类及选用 1.焊接设备&#xff08;对应焊接方法&#xff09; 2.焊接材料&#xff08;焊条、焊丝、焊剂、焊接气…

Ⅰ.计算机二级选择题(C语言概述)

【注&#xff1a;重点题以及添加目录格式导航&#xff01;&#xff01;&#xff01;】 【重点】&#xff08;第2题&#xff09; 【重点】&#xff08;第8题&#xff09; 【重点】&#xff08;第17题&#xff09; 【重点】&#xff08;第19题&#xff09; 【重点】&#xff08;第…

ck-editor5的研究 (4):初步使用 CKEditor5 的插件功能

前言 在上一篇文章中—— ck-editor5 的研究&#xff08;3&#xff09;&#xff1a;初步使用 CKEditor5 的事件系统和API &#xff0c;我们已经初步了解了 CKEditor5 的工作方式 那么这篇文章&#xff0c;我们将初步使用 CKEditor5 的插件功能&#xff0c;我将会写一个自己的…

分类预测 | Matlab实现CNN-LSTM-Attention高光谱数据分类

分类预测 | Matlab实现CNN-LSTM-Attention高光谱数据分类 目录 分类预测 | Matlab实现CNN-LSTM-Attention高光谱数据分类分类效果功能概述程序设计参考资料 分类效果 功能概述 代码功能 该MATLAB代码实现了一个结合CNN、LSTM和注意力机制的高光谱数据分类模型&#xff0c;核心…

嵌入式项目之mini2440系统制作烧写

系统移植的必要性与组成 在嵌入式开发中&#xff0c;**系统移植&#xff08;Linux 系统定制&#xff09;** 是常见的需求&#xff0c;主要原因在于&#xff1a; 1. **官方镜像体积过大**&#xff1a;标准 Linux 发行版&#xff08;如 Ubuntu&#xff09;可能占用数 GB 存储…

OpenShift AI - 启用过时版本的 Notebook 镜像

《OpenShift / RHEL / DevSecOps 汇总目录》 说明&#xff1a;本文已经在 OpenShift 4.18 OpenShift AI 2.19 的环境中验证 文章目录 查看可用 Notebook 镜像控制台查看命令行查看 Notebook 镜像和 Image Stream 对应关系启用老版本的 Notebook 镜像参考 查看可用 Notebook 镜…

“人单酬“理念:财税行业的自我驱动革命

引言&#xff1a;当薪酬不再是"固定数字"&#xff0c;而是"成长标尺" "为什么有人拼命工作却收入停滞&#xff1f;为什么企业总在人才流失中挣扎&#xff1f;"这些问题背后&#xff0c;往往隐藏着传统薪酬体系的僵化。而"人单酬"&…

yolo目标检测助手:具有模型预测、图像标注功能

在人工智能浪潮席卷各行各业的今天&#xff0c;计算机视觉模型&#xff08;如 YOLO&#xff09;已成为目标检测领域的标杆。然而&#xff0c;模型的强大能力需要直观的界面和便捷的工具才能充分发挥其演示、验证与迭代优化的价值。为此&#xff0c;我开发了一款基于 WPF 的桌面…

Spring Ai 从Demo到搭建套壳项目(一)初识与实现与deepseek对话模式

前言 为什么说Java长青&#xff0c;主要是因为其生态圈完善&#xff0c;Spring又做了一款脚手架&#xff0c;把对接各个LLM厂商的sdk做了一遍&#xff0c;形成一系列的spring-ai-starter-** 的依赖。 目前为止版本去到1.0.0.M6&#xff0c;golang跟不上了吧&#xff0c; Make …

机器学习实验七--SVM垃圾邮件分类器

SVM垃圾邮件分类器 一、什么是SVM二、实例&#xff1a;垃圾邮件分类器1.实验要求2.原理解释2.1 数据预处理流程2.2 特征提取方法2.3 SVM分类器 3.代码实现4.实验结果5.实验总结 一、什么是SVM 支持向量机(Support Vector Machine, SVM)是一种监督学习算法&#xff0c;主要用于…

lidar和imu的标定(一)Robust Real-time LiDAR-inertial Initialization

一、Robust Real-time LiDAR-inertial Initialization 看了这篇文章。在方法中&#xff0c;A和B都不细看了。主要看后边的几个部分。 C. LiDAR-inertial Initialization 在这一部分中&#xff0c; 1) Data Preprocess:主要是准备数据。 1.雷达里程计之后&#xff0c;可以得…

【手写系列】手写线程池

PS&#xff1a;本文的线程池为演示 Demo&#xff0c;皆在理解线程池的工作原理&#xff0c;并没有解决线程安全问题。 最简单一版的线程池 public class MyThreadPool {// 存放线程&#xff0c;复用已创建的线程List<Thread> threadList new ArrayList<>();publ…

Git企业级项目管理实战

目录 1. 准备工作 2. 添加成员 2.1 添加企业成员 2.2 添加项目成员 2.3 添加仓库开发人员 3. 开发场景 - 基于git flow模型的实践 3.1 新需求加入 3.2 修复测试环境 Bug 3.3 修改预发布环境Bug 3.4 修改正式环境 Bug 3.5 紧急修复正式环境 Bug 4. 拓展阅读 4.1 其…

go环境配置

下载对应版本的 go 版本 https://go.dev/dl/ 配置 vim ~/.zshrc export GOROOT/usr/local/go export PATH$PATH:$GOROOT/binsource ~/.zshrc >>>>>> go versiongoland 配置&#xff1a; &#x1f50d; 一、什么是GOPATH&#xff1f; GOPATH 是旧的项目结…

MySql(十二)

目录 MySql约束 1.添加主键约束 语法格式 1&#xff09;创建一个带主键的表 查看表结构 2&#xff09;创建表的时候指定主键名称 查看表结构 3&#xff09;创建一个表然后&#xff0c;然后再使用alter为列添加主键 查看表结构 4&#xff09;为表添加数据 1---正常数据 2---主键…

chrome.runtime.sendMessage 和 new FormData()

chrome.runtime.sendMessage 是Chrome扩展程序API中的一个方法&#xff0c;可用于背景脚本和内容脚本之间的消息传递。 new FormData() 提供了一种方便的方式来构建表单数据集。 在Chrome插件中&#xff0c;在 background.js 和 content.js 进行通信时使用了使用new FormData()…

数据结构-排序-排序的七种算法(2)

一&#xff0c;七种算法的介绍和比较 二&#xff0c;冒泡排序 原理&#xff1a;重复遍历列表&#xff0c;比较相邻元素&#xff0c;如果顺序错误就交换它们 时间复杂度&#xff1a; 最好&#xff1a;O(n)&#xff08;已有序时&#xff09; 平均&#xff1a;O(n) 最坏&#x…

【目标检测】backbone究竟有何关键作用?

backbone的核心在于能为检测提供若干种感受野大小和中心步长的组合&#xff0c;以满足对不同尺度和类别的目标检测。

JAVA实战开源项目:精简博客系统 (Vue+SpringBoot) 附源码

本文项目编号 T 215 &#xff0c;文末自助获取源码 \color{red}{T215&#xff0c;文末自助获取源码} T215&#xff0c;文末自助获取源码 目录 一、系统介绍二、数据库设计三、配套教程3.1 启动教程3.2 讲解视频3.3 二次开发教程 四、功能截图五、文案资料5.1 选题背景5.2 国内…