c++图论(二)之图的存储图解
在 C++ 中实现图的存储时,常用的方法包括 邻接矩阵(Adjacency Matrix)、邻接表(Adjacency List) 和 边列表(Edge List)。以下是具体实现方法、优缺点分析及代码示例:
1. 邻接矩阵(Adjacency Matrix)
原理
- 使用二维数组
matrix[u][v]表示顶点u和v的连接关系。 - 适用于 稠密图(边数接近顶点数的平方)。
- 无权图:
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. 动态图的存储优化
- 邻接表的动态扩展:使用
vector的push_back动态添加边。 - 删除边的优化:使用链表(如
list)或标记法(惰性删除)。
总结
- 邻接矩阵:适合稠密图,快速查询边的存在性。
- 邻接表:适合稀疏图,高效遍历邻接顶点(推荐使用
vector<vector<pair<int, int>>>)。 - 边列表:适合需要全局处理边的场景(如 Kruskal 算法)。
- 链式前向星:适合算法竞赛中的高性能需求。
代码建议:大多数情况下优先使用 邻接表,结合 C++ 的 vector 和 pair 实现带权图的高效存储。

相关文章:
c++图论(二)之图的存储图解
在 C 中实现图的存储时,常用的方法包括 邻接矩阵(Adjacency Matrix)、邻接表(Adjacency List) 和 边列表(Edge List)。以下是具体实现方法、优缺点分析及代码示例: 1. 邻接矩阵&…...
c++图论(一)之图论的起源和图的概念
C 图论之图论的起源和图的概念 图论(Graph Theory)是数学和计算机科学中的一个重要分支,其起源可以追溯到 18 世纪 的经典问题。以下是图论的历史背景、核心起源问题及其与基本概念和用途: 借用一下CSDN的图片哈 一、图论的起源&…...
《Python深度学习》第二讲:深度学习的数学基础
本讲来聊聊深度学习的数学基础。 深度学习听起来很厉害,其实它背后是一些很有趣的数学原理。本讲会用简单的方式解释这些原理,还会用一些具体的例子来帮助你理解。 2.1 初识神经网络 先从一个简单的任务开始:识别手写数字。 想象一下,你有一堆手写数字的图片,你想让计算…...
ChatGPT and Claude国内使用站点
RawChat kelaode chatgptplus chatopens(4.o mini免费,plus收费) 网页: 定价: wildcard 网页: 虚拟卡定价: 2233.ai 网页: 定价: MaynorAPI chatgpt cla…...
进行性核上性麻痹:精心护理,点亮希望之光
进行性核上性麻痹是一种罕见的神经退行性疾病,严重影响患者的生活质量。有效的健康护理能够在一定程度上缓解症状、延缓病情发展,给患者带来更好的生活体验。 在日常生活护理方面,由于患者平衡能力逐渐下降,行动不便,居…...
ZED X系列双目3D相机的耐用性与创新设计解析
在工业自动化和学术研究领域,高精度的视觉设备正成为提升效率和质量的关键。ZED X系列AI立体相机,凭借其先进的技术和耐用的设计,为这一领域带来了新的可能。 核心技术:深度感知与精准追踪 ZED X系列的核心技术之一是Neural Dept…...
HarmonyOS三层架构实战
目录: 1、三层架构项目结构1.0、三层架构简介1.1、 common层(主要放一些公共的资源等)1.2、 features层(主要模块定义的组件以及图片等静态资源)1.3、 products层(主要放主页面层和一些主要的资源ÿ…...
计算机四级 - 数据库原理 - 第4章 「关系数据库标准语言SQL」
4.1 SQL概述 4.1.1 结构化查询语言SQL SQL(Structured Query Language)称为结构化查询语言,它是由1974年由Boyce和Chamberi提出的,1975年至1979年IBM公司的San Jose Research Laboratory研制了关系数据库管理系统的原型系统System R,并实现了这种语198…...
基于PMU的14节点、30节点电力系统状态估计MATLAB程序
“电气仔推送”获得资料(专享优惠) 程序简介: 程序采用三种方法对14节点和30节点电力系统状态进行评估: ①PMU同步向量测量单元结合加权最小二乘法(WLS)分析电力系统的电压幅值和相角状态; …...
JS超过Number的最大值
场景:用户输入(这个可以通过前端限制输入长度控制)或正规场景,大数据量展示 Number类型的最大值是2^53 - 1 解决方案一:BigInt BigInt 是 JavaScript 中专门用来表示任意精度整数的类型。它允许你处理超出 Number 范围的整数。 const bigNu…...
Deepseek API+Python测试用例一键生成与导出-V1.0.2【实现需求文档图片识别与用例生成自动化】
在测试工作中,需求文档中的图片(如界面设计图、流程图)往往是测试用例生成的重要参考。然而,手动提取图片并识别内容不仅耗时,还容易出错。本文将通过一个自研小工具,结合 PaddleOCR 和大模型,自…...
整形在内存中的存储(例题逐个解析)
目录 一.相关知识点 1.截断: 2.整形提升: 3.如何 截断,整型提升? (1)负数 (2)正数 (3)无符号整型,高位补0 注意:提升后得到的…...
基于变分推理与 Best‑of‑N 策略的元 Prompt 自动生成与优化框架
摘要 本文提出了一种融合变分推理与 Best‑of‑N 策略的元 Prompt 自动生成与优化框架,通过高度参数化的模板、随机扰动采样及多指标评分机制,实现从初始提示生成到最终输出的动态优化。同时,针对实际应用中对自适应参数调整、深层语义理解、…...
AI 技术在智慧农业中的应用实践
智慧农业是通过现代信息技术(如物联网、大数据、人工智能等)提升农业生产效率、降低资源消耗、改善农产品质量的现代农业模式。AI 技术在智慧农业中的应用实践涵盖了从种植到收获的全流程,以下是具体的方案和应用场景: 1. AI 在智慧农业中的应用场景 1.1 精准种植 应用场景…...
蓝牙系统的核心组成解析
一、硬件层:看得见的物理载体 1. 射频模块(Radio Frequency Module) 专业描述:工作在2.4GHz ISM频段,支持GFSK/π/4 DQPSK/8DPSK调制方式 功能类比:相当于人的"嘴巴"和"耳朵" 发射端…...
centos 7误删/bash 拯救方法
进入救援模式 1. 插入CentOS 7安装光盘,重启系统。在开机时按BIOS设置对应的按键(通常是F2等),将启动顺序调整为CD - ROM优先。 2. 系统从光盘启动后,选择“Troubleshooting”,然后选择“Rescue a Cent…...
uniapp笔记-底部和首部标签页菜单生成
逻辑 这些都是需要配置pages.json文件。 其中底部需要手动配置tarBar,如: "tabBar": {"list":[{"pagePath": "pages/index/index","text": "首页"},{"pagePath": "pages/…...
基于Gemini 生成 Gemini Embedding
在本报告中,我们介绍了Gemini Embedding,这是一款基于谷歌功能最强大的大型语言模型Gemini的先进嵌入模型。借助Gemini的多语言和代码理解能力,Gemini Embedding能够为多种语言和文本模态的文本生成高度通用的嵌入表示。Gemini Embedding生成的表示可以预先计算并应用于多种…...
SpringBoot 和vue前后端配合开发网页拼图10关游戏源码技术分享
今天分享一个 前后端结合 的网页游戏 开发项目源码技术。 这也是我第一次写游戏类的程序,虽然不是特别复杂的游戏,但是是第一次写,肯定要记录一下了,哈哈。 游戏的内容 就是 我们显示中玩的那个 拼图碎片的 游戏,类似下…...
OpenCV计算摄影学(21)非真实感渲染之边缘保留滤波器edgePreservingFilter()
操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 算法描述 滤波是图像和视频处理中的基础操作。边缘保留平滑滤波器被广泛应用于多种不同场景[98]。 cv::edgePreservingFilter 是一种边缘保留滤波器&#…...
Qemu 详解与 ARM 虚拟机搭建指南
1. Qemu 是什么? Qemu(Quick Emulator)是一款开源的机器模拟器和虚拟化工具,支持多种硬件架构(如 x86、ARM、PowerPC 等)。它的核心功能包括: 动态指令翻译:将不同架构的指令实时翻…...
JVM并发编程AQSsync锁ReentrantLock线程池ThreadLocal
并发编程2 synchronized锁实现**AQS****ReentrantLock实现****JUC 常用类**池的概念 ThreadLocalThreadLocal原理内存泄露强引用:软引用弱引用虚引用ThreadLocal内存泄露 synchronized锁实现 synchronized是一个关键字,实现同步,还需要我们提供一个同步锁对象,记录锁状态,记录…...
CMake学习笔记(三):静态库,动态库的生成和使用
一:动态库 接下来我们简单的讲解下动态库的建立和使用:在后面的项目的开发过程中,我们使用第三方库或者我们跑这个项目的时候我们总会看到一些.so的文件,这些就是所谓的动态库,里面的内容就是编译后的源文件,是程序运行时被加载和…...
《Classifier-Free Diffusion Guidance》的核心观点与方法
介绍《Classifier-Free Diffusion Guidance》的核心观点与方法 在扩散模型(Diffusion Models)的研究中,如何在生成样本的质量与多样性之间找到平衡一直是核心挑战之一。传统的生成模型(如GANs或Glow)通过截断…...
什么是数学建模?数学建模是将实际问题转化为数学问题
数学建模是将实际问题转化为数学问题,并通过数学工具进行分析、求解和验证的过程。 一、数学建模的基本流程 问题分析 • 明确目标:确定需要解决的核心问题。 • 简化现实:识别关键变量、忽略次要因素。 • 定义输入和输出:明确模…...
唤起“队列”的回忆
又来博客记录自己的学习心得了,嘿嘿嘿(^~^) 目录 队列的概念和结构: 队列的创建和初始化: 队列入栈: 队列出栈: 队列的销毁: 取队头和队尾数据: 结语: 队列的概念…...
Linux(8.4)NFS
文章目录 一、概念二、详解NFS1)软件名2)服务名3)配置文件4)端口号5)相关命令 三、部署NFS一、NFS服务端1)**配置源(本地或者网络源)**2)2、安装NFS**3)启动服…...
【位运算】速算密钥:位运算探秘
文章目录 前言例题一、判定字符是否唯一二、丢失的数字三、两整数之和四、只出现⼀次的数字 II五、消失的两个数字 结语 前言 什么是位运算算法呢? 位运算算法是以位运算为核心操作,设计用来高效解决特定问题的一系列计算步骤集合。它巧妙利用位运算直接…...
STM32G070CBT6读写FLASH中的数据
向FLASH中写入数据函数 /*函数说明:向FLASH中写数据形参:addr-要写入数据的起始地址 data-准备写入数据 len-数据大小返回值:1-成功,0-失败 */ uint8_t FlashWriteData(uint64_t addr,uint8_t data[],size_t len) {uint32_t Fir…...
算法刷题记录——LeetCode篇(4) [第301~400题](持续更新)
(优先整理热门100及面试150,不定期持续更新,欢迎关注) 322. 零钱兑换 给你一个整数数组 coins ,表示不同面额的硬币;以及一个整数 amount ,表示总金额。 计算并返回可以凑成总金额所需的最少的硬币个数。如果没有任何…...
