c++图论(二)之图的存储图解

news2025/3/18 13:23:34

在 C++ 中实现图的存储时,常用的方法包括 邻接矩阵(Adjacency Matrix)邻接表(Adjacency List)边列表(Edge List)。以下是具体实现方法、优缺点分析及代码示例:


1. 邻接矩阵(Adjacency Matrix)

原理
  • 使用二维数组 matrix[u][v] 表示顶点 uv 的连接关系。
  • 适用于 稠密图(边数接近顶点数的平方)。
  • 无权图matrix[u][v] = 1 表示存在边;0 表示无连接。
  • 带权图matrix[u][v] = weight 表示边的权重。
图解

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

C++ 实现
#include <vector>
using namespace std;

// 定义图的顶点数
const int V = 100;

// 无权图的邻接矩阵
vector<vector<int>> adjMatrix(V, vector<int>(V, 0));

// 添加无向边
void addUndirectedEdge(int u, int v) {
    adjMatrix[u][v] = 1;
    adjMatrix[v][u] = 1;
}

// 添加带权有向边
void addDirectedWeightedEdge(int u, int v, int weight) {
    adjMatrix[u][v] = weight;
}

// 检查边是否存在
bool hasEdge(int u, int v) {
    return adjMatrix[u][v] != 0;
}
优点
  • 快速判断两顶点是否相邻:时间复杂度 O(1)。
  • 适合频繁查询边的存在性
缺点
  • 空间复杂度高:O(V²),不适合顶点数多(如 V > 1e4)的稀疏图。
  • 插入/删除边效率低:需要修改二维数组。

2. 邻接表(Adjacency List)

原理
  • 为每个顶点维护一个链表或动态数组,存储其邻接顶点。
  • 适用于 稀疏图(边数远小于顶点数的平方)。
  • 无权图adjList[u] 存储 v 的集合。
  • 带权图adjList[u] 存储 pair<v, weight>
C++ 实现
#include <vector>
#include <list>
using namespace std;

// 无权图的邻接表(使用 vector)
vector<vector<int>> adjList;

// 初始化顶点数为 n 的图
void initGraph(int n) {
    adjList.resize(n);
}

// 添加无向边
void addUndirectedEdge(int u, int v) {
    adjList[u].push_back(v);
    adjList[v].push_back(u);
}

// 带权图的邻接表(使用 vector<pair>)
vector<vector<pair<int, int>>> weightedAdjList;

// 添加带权有向边
void addWeightedDirectedEdge(int u, int v, int weight) {
    weightedAdjList[u].emplace_back(v, weight); // C++11 的 emplace_back 更高效
}

// 遍历顶点 u 的邻居
void traverseNeighbors(int u) {
    for (const auto& neighbor : adjList[u]) {
        // 处理邻居顶点 neighbor
    }
}
优点
  • 空间复杂度低:O(V + E),适合大规模稀疏图。
  • 高效遍历邻接顶点:时间复杂度与邻接顶点数成正比。
缺点
  • 查询边的存在性慢:需要遍历邻接表,时间复杂度 O(degree(u))。

3. 边列表(Edge List)

原理
  • 将图的边存储为 (u, v, weight) 的列表。
  • 适用于需要 按边遍历 的场景(如 Kruskal 算法求最小生成树)。
C++ 实现
#include <vector>
using namespace std;

// 定义边的结构体
struct Edge {
    int u, v, weight;
    Edge(int u, int v, int w) : u(u), v(v), weight(w) {}
};

vector<Edge> edgeList;

// 添加带权边
void addEdge(int u, int v, int weight) {
    edgeList.emplace_back(u, v, weight);
}

// 遍历所有边
void traverseEdges() {
    for (const Edge& e : edgeList) {
        // 处理边 e.u -> e.v,权重 e.weight
    }
}
优点
  • 存储简单:适用于算法需要全局遍历边(如 Kruskal 算法)。
  • 节省空间:仅存储存在的边,空间复杂度 O(E)。
缺点
  • 查询顶点邻接关系慢:需要遍历整个边列表。

4. 链式前向星(Linked Forward Star)

原理
  • 一种紧凑的邻接表实现,通过数组模拟链表,常用于算法竞赛。
  • 使用三个数组:head[]to[]next[]weight[]
C++ 实现
const int MAX_EDGES = 1e5; // 最大边数
int head[MAX_EDGES];       // head[u] 表示顶点 u 的第一条边的索引
int to[MAX_EDGES];         // 存储边的终点
int next[MAX_EDGES];       // 存储下一条边的索引
int weight[MAX_EDGES];     // 存储边的权重
int edgeCount = 0;         // 当前边数

// 初始化
void init() {
    memset(head, -1, sizeof(head)); // 初始化为 -1
}

// 添加有向边 u -> v,权重 w
void addEdge(int u, int v, int w) {
    to[edgeCount] = v;
    weight[edgeCount] = w;
    next[edgeCount] = head[u];
    head[u] = edgeCount++;
}

// 遍历顶点 u 的邻接边
void traverseEdges(int u) {
    for (int i = head[u]; i != -1; i = next[i]) {
        int v = to[i];
        int w = weight[i];
        // 处理边 u -> v,权重 w
    }
}
优点
  • 内存紧凑:适合处理超大规模图(如顶点数 1e5 以上)。
  • 高效遍历:与邻接表性能接近。
缺点
  • 实现复杂:需要手动管理数组索引。

5. 存储方法对比及适用场景

存储方法时间复杂度(查询边)空间复杂度适用场景
邻接矩阵O(1)O(V²)稠密图、频繁查询边的存在性
邻接表O(degree(u))O(V + E)稀疏图、频繁遍历邻接顶点
边列表O(E)O(E)需要全局遍历边的算法
链式前向星O(degree(u))O(V + E)算法竞赛中的大规模图处理

6. 动态图的存储优化

  • 邻接表的动态扩展:使用 vectorpush_back 动态添加边。
  • 删除边的优化:使用链表(如 list)或标记法(惰性删除)。

总结

  • 邻接矩阵:适合稠密图,快速查询边的存在性。
  • 邻接表:适合稀疏图,高效遍历邻接顶点(推荐使用 vector<vector<pair<int, int>>>)。
  • 边列表:适合需要全局处理边的场景(如 Kruskal 算法)。
  • 链式前向星:适合算法竞赛中的高性能需求。

代码建议:大多数情况下优先使用 邻接表,结合 C++ 的 vectorpair 实现带权图的高效存储。


在这里插入图片描述

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

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

相关文章

c++图论(一)之图论的起源和图的概念

C 图论之图论的起源和图的概念 图论&#xff08;Graph Theory&#xff09;是数学和计算机科学中的一个重要分支&#xff0c;其起源可以追溯到 18 世纪 的经典问题。以下是图论的历史背景、核心起源问题及其与基本概念和用途&#xff1a; 借用一下CSDN的图片哈 一、图论的起源&…

ChatGPT and Claude国内使用站点

RawChat kelaode chatgptplus chatopens&#xff08;4.o mini免费&#xff0c;plus收费&#xff09; 网页&#xff1a; 定价&#xff1a; wildcard 网页&#xff1a; 虚拟卡定价&#xff1a; 2233.ai 网页&#xff1a; 定价&#xff1a; MaynorAPI chatgpt cla…

进行性核上性麻痹:精心护理,点亮希望之光

进行性核上性麻痹是一种罕见的神经退行性疾病&#xff0c;严重影响患者的生活质量。有效的健康护理能够在一定程度上缓解症状、延缓病情发展&#xff0c;给患者带来更好的生活体验。 在日常生活护理方面&#xff0c;由于患者平衡能力逐渐下降&#xff0c;行动不便&#xff0c;居…

ZED X系列双目3D相机的耐用性与创新设计解析

在工业自动化和学术研究领域&#xff0c;高精度的视觉设备正成为提升效率和质量的关键。ZED X系列AI立体相机&#xff0c;凭借其先进的技术和耐用的设计&#xff0c;为这一领域带来了新的可能。 核心技术&#xff1a;深度感知与精准追踪 ZED X系列的核心技术之一是Neural Dept…

HarmonyOS三层架构实战

目录&#xff1a; 1、三层架构项目结构1.0、三层架构简介1.1、 common层&#xff08;主要放一些公共的资源等&#xff09;1.2、 features层&#xff08;主要模块定义的组件以及图片等静态资源&#xff09;1.3、 products层&#xff08;主要放主页面层和一些主要的资源&#xff…

计算机四级 - 数据库原理 - 第4章 「关系数据库标准语言SQL」

4.1 SQL概述 4.1.1 结构化查询语言SQL SQL(Structured Query Language)称为结构化查询语言&#xff0c;它是由1974年由Boyce和Chamberi提出的&#xff0c;1975年至1979年IBM公司的San Jose Research Laboratory研制了关系数据库管理系统的原型系统System R,并实现了这种语198…

基于PMU的14节点、30节点电力系统状态估计MATLAB程序

“电气仔推送”获得资料&#xff08;专享优惠&#xff09; 程序简介&#xff1a; 程序采用三种方法对14节点和30节点电力系统状态进行评估&#xff1a; ①PMU同步向量测量单元结合加权最小二乘法&#xff08;WLS&#xff09;分析电力系统的电压幅值和相角状态&#xff1b; …

Deepseek API+Python测试用例一键生成与导出-V1.0.2【实现需求文档图片识别与用例生成自动化】

在测试工作中&#xff0c;需求文档中的图片&#xff08;如界面设计图、流程图&#xff09;往往是测试用例生成的重要参考。然而&#xff0c;手动提取图片并识别内容不仅耗时&#xff0c;还容易出错。本文将通过一个自研小工具&#xff0c;结合 PaddleOCR 和大模型&#xff0c;自…

整形在内存中的存储(例题逐个解析)

目录 一.相关知识点 1.截断&#xff1a; 2.整形提升&#xff1a; 3.如何 截断&#xff0c;整型提升&#xff1f; &#xff08;1&#xff09;负数 &#xff08;2&#xff09;正数 &#xff08;3&#xff09;无符号整型&#xff0c;高位补0 注意&#xff1a;提升后得到的…

蓝牙系统的核心组成解析

一、硬件层&#xff1a;看得见的物理载体 1. 射频模块&#xff08;Radio Frequency Module&#xff09; 专业描述&#xff1a;工作在2.4GHz ISM频段&#xff0c;支持GFSK/π/4 DQPSK/8DPSK调制方式 功能类比&#xff1a;相当于人的"嘴巴"和"耳朵" 发射端…

uniapp笔记-底部和首部标签页菜单生成

逻辑 这些都是需要配置pages.json文件。 其中底部需要手动配置tarBar&#xff0c;如&#xff1a; "tabBar": {"list":[{"pagePath": "pages/index/index","text": "首页"},{"pagePath": "pages/…

SpringBoot 和vue前后端配合开发网页拼图10关游戏源码技术分享

今天分享一个 前后端结合 的网页游戏 开发项目源码技术。 这也是我第一次写游戏类的程序&#xff0c;虽然不是特别复杂的游戏&#xff0c;但是是第一次写&#xff0c;肯定要记录一下了&#xff0c;哈哈。 游戏的内容 就是 我们显示中玩的那个 拼图碎片的 游戏&#xff0c;类似下…

OpenCV计算摄影学(21)非真实感渲染之边缘保留滤波器edgePreservingFilter()

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 滤波是图像和视频处理中的基础操作。边缘保留平滑滤波器被广泛应用于多种不同场景[98]。 cv::edgePreservingFilter 是一种边缘保留滤波器&#…

JVM并发编程AQSsync锁ReentrantLock线程池ThreadLocal

并发编程2 synchronized锁实现**AQS****ReentrantLock实现****JUC 常用类**池的概念 ThreadLocalThreadLocal原理内存泄露强引用:软引用弱引用虚引用ThreadLocal内存泄露 synchronized锁实现 synchronized是一个关键字,实现同步,还需要我们提供一个同步锁对象,记录锁状态,记录…

什么是数学建模?数学建模是将实际问题转化为数学问题

数学建模是将实际问题转化为数学问题&#xff0c;并通过数学工具进行分析、求解和验证的过程。 一、数学建模的基本流程 问题分析 • 明确目标&#xff1a;确定需要解决的核心问题。 • 简化现实&#xff1a;识别关键变量、忽略次要因素。 • 定义输入和输出&#xff1a;明确模…

唤起“队列”的回忆

又来博客记录自己的学习心得了&#xff0c;嘿嘿嘿(^&#xff5e;^) 目录 队列的概念和结构&#xff1a; 队列的创建和初始化&#xff1a; 队列入栈&#xff1a; 队列出栈&#xff1a; 队列的销毁&#xff1a; 取队头和队尾数据&#xff1a; 结语&#xff1a; 队列的概念…

Linux(8.4)NFS

文章目录 一、概念二、详解NFS1&#xff09;软件名2&#xff09;服务名3&#xff09;配置文件4&#xff09;端口号5&#xff09;相关命令 三、部署NFS一、NFS服务端1&#xff09;**配置源&#xff08;本地或者网络源&#xff09;**2&#xff09;2、安装NFS**3&#xff09;启动服…

【位运算】速算密钥:位运算探秘

文章目录 前言例题一、判定字符是否唯一二、丢失的数字三、两整数之和四、只出现⼀次的数字 II五、消失的两个数字 结语 前言 什么是位运算算法呢&#xff1f; 位运算算法是以位运算为核心操作&#xff0c;设计用来高效解决特定问题的一系列计算步骤集合。它巧妙利用位运算直接…

STM32G070CBT6读写FLASH中的数据

向FLASH中写入数据函数 /*函数说明&#xff1a;向FLASH中写数据形参&#xff1a;addr-要写入数据的起始地址 data-准备写入数据 len-数据大小返回值&#xff1a;1-成功&#xff0c;0-失败 */ uint8_t FlashWriteData(uint64_t addr,uint8_t data[],size_t len) {uint32_t Fir…

AD绘图基本操作

一、基本操作 注意&#xff1a;快捷键都要在英文模式下才能生效 1、移动 按住鼠标右键移动 2、切换桌面栅格距离 G 3、英寸和毫米 尺寸切换 Q 4、元件在3D模式下的移动 3D视角鼠标左键只起到选择元器件并移动之的功能&#xff0c; 单纯鼠标右键只能平移桌面 shift鼠…