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

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

相关文章

java实现docker镜像上传到harbor仓库的方式

《java实现docker镜像上传到harbor仓库的方式》:本文主要介绍java实现docker镜像上传到harbor仓库的方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 前 言2. 编写工具类2.1 引入依赖包2.2 使用当前服务器的docker环境推送镜像2.2

Java中调用数据库存储过程的示例代码

《Java中调用数据库存储过程的示例代码》本文介绍Java通过JDBC调用数据库存储过程的方法,涵盖参数类型、执行步骤及数据库差异,需注意异常处理与资源管理,以优化性能并实现复杂业务逻辑,感兴趣的朋友... 目录一、存储过程概述二、Java调用存储过程的基本javascript步骤三、Java调用存储过程示

MySQL之InnoDB存储引擎中的索引用法及说明

《MySQL之InnoDB存储引擎中的索引用法及说明》:本文主要介绍MySQL之InnoDB存储引擎中的索引用法及说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录1、背景2、准备3、正篇【1】存储用户记录的数据页【2】存储目录项记录的数据页【3】聚簇索引【4】二

MySQL之InnoDB存储页的独立表空间解读

《MySQL之InnoDB存储页的独立表空间解读》:本文主要介绍MySQL之InnoDB存储页的独立表空间,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、背景2、独立表空间【1】表空间大小【2】区【3】组【4】段【5】区的类型【6】XDES Entry区结构【

SQLite3 在嵌入式C环境中存储音频/视频文件的最优方案

《SQLite3在嵌入式C环境中存储音频/视频文件的最优方案》本文探讨了SQLite3在嵌入式C环境中存储音视频文件的优化方案,推荐采用文件路径存储结合元数据管理,兼顾效率与资源限制,小文件可使用B... 目录SQLite3 在嵌入式C环境中存储音频/视频文件的专业方案一、存储策略选择1. 直接存储 vs

springboot项目打jar制作成镜像并指定配置文件位置方式

《springboot项目打jar制作成镜像并指定配置文件位置方式》:本文主要介绍springboot项目打jar制作成镜像并指定配置文件位置方式,具有很好的参考价值,希望对大家有所帮助,如有错误... 目录一、上传jar到服务器二、编写dockerfile三、新建对应配置文件所存放的数据卷目录四、将配置文

gitlab安装及邮箱配置和常用使用方式

《gitlab安装及邮箱配置和常用使用方式》:本文主要介绍gitlab安装及邮箱配置和常用使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1.安装GitLab2.配置GitLab邮件服务3.GitLab的账号注册邮箱验证及其分组4.gitlab分支和标签的

C++中零拷贝的多种实现方式

《C++中零拷贝的多种实现方式》本文主要介绍了C++中零拷贝的实现示例,旨在在减少数据在内存中的不必要复制,从而提高程序性能、降低内存使用并减少CPU消耗,零拷贝技术通过多种方式实现,下面就来了解一下... 目录一、C++中零拷贝技术的核心概念二、std::string_view 简介三、std::stri

Linux脚本(shell)的使用方式

《Linux脚本(shell)的使用方式》:本文主要介绍Linux脚本(shell)的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录概述语法详解数学运算表达式Shell变量变量分类环境变量Shell内部变量自定义变量:定义、赋值自定义变量:引用、修改、删

python判断文件是否存在常用的几种方式

《python判断文件是否存在常用的几种方式》在Python中我们在读写文件之前,首先要做的事情就是判断文件是否存在,否则很容易发生错误的情况,:本文主要介绍python判断文件是否存在常用的几种... 目录1. 使用 os.path.exists()2. 使用 os.path.isfile()3. 使用