再谈图的存储方式(邻接矩阵,邻接表,前向星)

2024-06-11 00:32

本文主要是介绍再谈图的存储方式(邻接矩阵,邻接表,前向星),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.邻接矩阵

1.存图思想

使用一个矩阵来描述一个图,对于矩阵的第i行第j列的值,表示编号为i的顶点到编号为j的顶点的权值。

2.代码实现

// 最大顶点数
const int V = 1000;// 邻接矩阵的定义
// mat[i][j] 表示 顶点'i'到顶点'j'的权值
int mat[V][V];// 邻接矩阵的初始化操作
// 假设权值为零表示没有该边
memset(mat, 0, sizeof(mat))// 增加边
// 新增顶点'i'到顶点'j'的边,权值为w
mat[i][j] = w;//遍历邻接边
for(int i=0;i<n;i++)
{for(int j=0;j<n;j++){if(mat[i][j]!=0)//doing something.}
}

2.邻接表

1.存图思想

邻接矩阵对于每个顶点使用定长的数组来存储以该点出发的边的情况。第i个数组的第j个值存储的是从顶点i到顶点j的边的权值。 
邻接表则是对于每个顶点使用不定长的链表来存储以该点出发的边的情况。因此对第i个链表的第j个值实际上存储的是编号为i的顶点出发的第j条边的情况。

2.代码实现

在ACM题目中,动态的数据结构一般是不被推荐的,因为动态开辟内存比较消耗时间,且写起来复杂容易出错。 
大部分情况我们使用C++STL里的vector作为链表来实现图的邻接表。
// 最大顶点数
const int V = 100000;// vector实现的邻接表的定义
// 不考虑边权,存储类型为int型
vector<int> e[V];
// 若考虑边权,则定义一个结构,vector也为结构体类型
struct node{int v,int w};//存储边的终点和边的权值
vector<node> e[V];
//也可以用一个数组或vector单独存边的信息,然后在邻接表vector中记录每个点邻接边的编号// 邻接表的初始化操作
for(int i=0;i<n;i++)
{e[i].clear();
}// 增加边
//不考虑边权
e[i].push_back(j);
//考虑边权
e[i].push_back(node(j,w));//遍历邻接边
for (int j=0; j<(int)e[i].size(); ++j) {int k=e[i][j];//第j条边为[i,k]//ornode &e=e[i][j];//第j条边为[i,e.v,e.w]// do something.
}

3.前向星

1.存图思想

前向星是一种特殊的边集数组,我们把边集数组中的每一条边按照起点从小到大排序,如果起点相同就按照终点从小到大排序,  并记录下以某个点为起点的所有边在数组中的起始位置和存储长度,那么前向星就构造好了. 
用len[i]来记录所有以i为起点的边在数组中的存储长度. //不用len数组也可以  
用head[i]记录以i为边集在数组中的第一个存储位置.

2.代码实现

一般不用不写了。。。

4.链式前向星

1.存图思想

我们建立边结构体为:
struct Edge
{int next;int to;int w;
};
其中edge[i].to表示第i条边的终点,edge[i].next表示与第i条边同起点的下一条边的存储位置,edge[i].w为边权值. 
另外还有一个数组head[],它是用来表示以i为起点的第一条边存储的位置,实际上你会发现这里的第一条边存储的位置其实是在以i为起点的所有边的最后输入的那个编号.head[]数组一般初始化为-1。 

2.代码实现

// 最大顶点数
const int V = 1000;
const int E = 10000;struct Edge
{int next;int to;int w;
};
Edge edge[E];
int head[V];//初始化
memset(head,0xff,sizeof(head));
memset(edge,0,sizeof(edge));//增加边
void add(int u,int v,int w) 
{  //cnt为边计数edge[cnt].w = w;  edge[cnt].to = v;  edge[cnt].next = head[u];  head[u] = cnt++;  
} //遍历边
for(int i=1;i<=n;i++)
{for(int k=head[i];k!=-1;k=edge[k].next){//doing something.}
}


相关资料:
1.链式前向星及其简单应用 | Malash's Blog
2.前向星与链式前向星 | 学步园
3.ACM图论之存图方式 | 剑紫青天

这篇关于再谈图的存储方式(邻接矩阵,邻接表,前向星)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1049704

相关文章

异构存储(冷热数据分离)

异构存储主要解决不同的数据,存储在不同类型的硬盘中,达到最佳性能的问题。 异构存储Shell操作 (1)查看当前有哪些存储策略可以用 [lytfly@hadoop102 hadoop-3.1.4]$ hdfs storagepolicies -listPolicies (2)为指定路径(数据存储目录)设置指定的存储策略 hdfs storagepolicies -setStoragePo

HDFS—存储优化(纠删码)

纠删码原理 HDFS 默认情况下,一个文件有3个副本,这样提高了数据的可靠性,但也带来了2倍的冗余开销。 Hadoop3.x 引入了纠删码,采用计算的方式,可以节省约50%左右的存储空间。 此种方式节约了空间,但是会增加 cpu 的计算。 纠删码策略是给具体一个路径设置。所有往此路径下存储的文件,都会执行此策略。 默认只开启对 RS-6-3-1024k

内核启动时减少log的方式

内核引导选项 内核引导选项大体上可以分为两类:一类与设备无关、另一类与设备有关。与设备有关的引导选项多如牛毛,需要你自己阅读内核中的相应驱动程序源码以获取其能够接受的引导选项。比如,如果你想知道可以向 AHA1542 SCSI 驱动程序传递哪些引导选项,那么就查看 drivers/scsi/aha1542.c 文件,一般在前面 100 行注释里就可以找到所接受的引导选项说明。大多数选项是通过"_

用命令行的方式启动.netcore webapi

用命令行的方式启动.netcore web项目 进入指定的项目文件夹,比如我发布后的代码放在下面文件夹中 在此地址栏中输入“cmd”,打开命令提示符,进入到发布代码目录 命令行启动.netcore项目的命令为:  dotnet 项目启动文件.dll --urls="http://*:对外端口" --ip="本机ip" --port=项目内部端口 例: dotnet Imagine.M

深入理解RxJava:响应式编程的现代方式

在当今的软件开发世界中,异步编程和事件驱动的架构变得越来越重要。RxJava,作为响应式编程(Reactive Programming)的一个流行库,为Java和Android开发者提供了一种强大的方式来处理异步任务和事件流。本文将深入探讨RxJava的核心概念、优势以及如何在实际项目中应用它。 文章目录 💯 什么是RxJava?💯 响应式编程的优势💯 RxJava的核心概念

【即时通讯】轮询方式实现

技术栈 LayUI、jQuery实现前端效果。django4.2、django-ninja实现后端接口。 代码仓 - 后端 代码仓 - 前端 实现功能 首次访问页面并发送消息时需要设置昵称发送内容为空时要提示用户不能发送空消息前端定时获取消息,然后展示在页面上。 效果展示 首次发送需要设置昵称 发送消息与消息展示 提示用户不能发送空消息 后端接口 发送消息 DB = []@ro

脏页的标记方式详解

脏页的标记方式 一、引言 在数据库系统中,脏页是指那些被修改过但还未写入磁盘的数据页。为了有效地管理这些脏页并确保数据的一致性,数据库需要对脏页进行标记。了解脏页的标记方式对于理解数据库的内部工作机制和优化性能至关重要。 二、脏页产生的过程 当数据库中的数据被修改时,这些修改首先会在内存中的缓冲池(Buffer Pool)中进行。例如,执行一条 UPDATE 语句修改了某一行数据,对应的缓

速了解MySQL 数据库不同存储引擎

快速了解MySQL 数据库不同存储引擎 MySQL 提供了多种存储引擎,每种存储引擎都有其特定的特性和适用场景。了解这些存储引擎的特性,有助于在设计数据库时做出合理的选择。以下是 MySQL 中几种常用存储引擎的详细介绍。 1. InnoDB 特点: 事务支持:InnoDB 是一个支持 ACID(原子性、一致性、隔离性、持久性)事务的存储引擎。行级锁:使用行级锁来提高并发性,减少锁竞争

Java 多线程的基本方式

Java 多线程的基本方式 基础实现两种方式: 通过实现Callable 接口方式(可得到返回值):

前端form表单+ifarme方式实现大文件下载

// main.jsimport Vue from 'vue';import App from './App.vue';import { downloadTokenFile } from '@/path/to/your/function'; // 替换为您的函数路径// 将 downloadTokenFile 添加到 Vue 原型上Vue.prototype.$downloadTokenF