图的存储 —— 邻接矩阵

这篇具有很好参考价值的文章主要介绍了图的存储 —— 邻接矩阵。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

图的结构比较复杂,任何两个节点之间都可能有关系。
图的存储分为顺序存储和链式存储。
顺序存储包括邻接矩阵和边集数组,
链式存储包括邻接表、链式前向星、十字链表和邻接多重表。

图的存储 —— 邻接矩阵

邻接矩阵通常采用一个一维数组存储图中节点的信息,采用一个二维数组存储图中节点之间的邻接关系。

【邻接矩阵的表示方法】

无向图、有向图和网的邻接矩阵的表示方法如下所述。

① 无向图的邻接矩阵

在无向图中,若从节点vi 到节点vj 有边,则邻接矩阵 M [ i ][ j ]= M [ j ][ i ]=1,否则 M [ i ][ j ]=0。

图的存储 —— 邻接矩阵

例如,一个无向图的节点信息和邻接矩阵如下图所示。

图的存储 —— 邻接矩阵

在该无向图中,从节点a 到节点b 有边,从节点b 到节点a 也有边,节点a 、b 在一维数组中的存储位置分别为0、1,则 M [ 0 ][ 1 ]= M [ 1 ][ 0 ]=1。

图的存储 —— 邻接矩阵

无向图的邻接矩阵的特点如下:

  • 无向图的邻接矩阵是对称矩阵,并且是唯一的。
  • 第i 行或第i 列非零元素的个数正好是第i 个节点的度。上图中的邻接矩阵,第3列非零元素的个数为2,说明第3个节点c 的度为2。

② 有向图的邻接矩阵

在有向图中,若从节点vi 到节点vj 有边,则邻接矩阵 M [ i ][ j ]=1,否则 M [ i ][ j ]=0。

图的存储 —— 邻接矩阵

以尖括号<vi ,vj >表示的是有序对,以圆括号(vi ,vj )表示的是无序对

例如,一个有向图的节点信息和邻接矩阵如下图所示。

图的存储 —— 邻接矩阵

在该有向图中,从节点a 到节点b 有边,节点a 、b 在一维数组中的存储位置分别为0、1,因此 M [ 0 ][ 1 ]=1。有向图中的边是有向边,从节点a 到节点b有边,从节点b 到节点a 不一定有边,因此有向图的邻接矩阵不一定是对称的。

图的存储 —— 邻接矩阵

有向图的邻接矩阵的特点如下:

  • 有向图的邻接矩阵不一定是对称的。
  • 第i 行非零元素的个数正好是第i 个节点的出度,第i 列非零元素的个数正好是第i 个节点的入度。上图中的邻接矩阵,第3行非零元素的个数为2,第3列非零元素的个数也为2,说明第3个节点c 的出度和入度均为2。

③ 网的邻接矩阵

网是带权图,需要存储边的权值,则邻接矩阵表示为

图的存储 —— 邻接矩阵

其中,wij 表示边上的权值,∞表示无穷大。当i =j 时,wii 也可被设置为0。

例如,一个网的节点信息和邻接矩阵如下图所示。

图的存储 —— 邻接矩阵

在该网中,从节点a 到节点b 有边,且该边的权值为2,节点a 、b 在一维数组中的存储位置分别为0、1,因此 M [ 0 ][ 1 ]=2。从节点b 到节点a 没有边,因此 M [ 1 ][ 0 ]=∞。

图的存储 —— 邻接矩阵

【邻接矩阵的数据结构定义】

首先定义邻接矩阵的数据结构,如下图所示。

图的存储 —— 邻接矩阵

#define MaxVnum 100  //节点数的最大值
typedef char VexType;  //节点的数据类型,根据需要定义
typedef int EdgeType;  //边上权值的数据类型,若为不带权值的图,则为0或1

【邻接矩阵的存储方法】

[算法步骤]

  1. 输入节点数和边数
  2. 依次输入节点信息,将其存储到节点数组Vex[]中;
  3. 初始化邻接矩阵,如果是图,则将其初始化为0;如果是网,则将其初始化为∞;
  4. 依次输入每条边依附的两个节点,如果是网,则还需要输入该边的权值。
  • 如果是无向图,则输入a b ,查询节点a、b 在节点数组Vex[]中的存储下标i 、j ,令Edge[ i ][ j ]=Edge[ j ][ i ]=1
  • 如果是无向网,则输入a b w ,查询节点a、b 在节点数组Vex[]中的存储下标i 、j ,令Edge[ i ][ j ]=Edge[ j ][ i ]=w 。
  • 如果是有向网,则输入a b w ,查询节点a、b 在节点数组Vex[]中的存储下标i 、j ,令Edge[ i ][ j ]=w 。

[完美图解]

一个无向图如下图所示,其邻接矩阵的存储过程如下所述。

图的存储 —— 邻接矩阵

① 输入节点数和边数。

4 5

结果:G .vexnum=4、G .edgenum=5。

② 输入节点信息,将其存入节点信息数组。

a b c d

存储结果如下图所示。

图的存储 —— 邻接矩阵

③ 初始化邻接矩阵的值均为0,如下图所示。

图的存储 —— 邻接矩阵

④ 依次输入每条边依附的两个节点。

  • 输入a b ,处理结果:在Vex[]数组中查找到节点a 、b 的下标分别为0、1,是无向图,因此令Edge[0][1]=Edge[1][0]=1,如下图所示。
    图的存储 —— 邻接矩阵

  • 输入a d ,处理结果:在Vex[]数组中查找到节点a 、d 的下标分别为0、3,是无向图,因此令Edge[0][3]= Edge[3][0]=1,如下图所示。
    图的存储 —— 邻接矩阵

  • 输入b c ,处理结果:在Vex[]数组中查找到节点b 、c 的下标分别为1、2,是无向图,因此令Edge[1][2]= Edge[2][1]=1,如下图所示
    图的存储 —— 邻接矩阵

  • 输入b d ,处理结果:在Vex[]数组中查找到节点b 、d 的下标分别为1、3,是无向图,因此令Edge[1][3]= Edge[3][1]=1,如下图所示
    图的存储 —— 邻接矩阵

  • 输入c d ,处理结果:在Vex[]数组中查找到节点c 、d 的下标分别为2、3,是无向图,因此令Edge[2][3]= Edge[3][2]=1,如下图所示
    图的存储 —— 邻接矩阵

在实际应用中,也可以先输入节点信息并将其存入数组Vex[]。在输入边时直接输入节点的存储下标序号,这样可以节省查询节点下标所需的时间,从而提高效率。

[算法实现代码]

void CreateAMGragh(AMGragh &G){
	int i , j ;
	VexType u , v;
	cout << "请输入节点数:" << endl;
	cin >> G.vexnum;
	cout << "请输入边数:" << endl;
	cin >> G.edgenum;

	cout << "请输入节点信息:" << endl;
	for(int i = 0 ; i < G.vexnum ; i ++){ //输入节点信息,将其存入节点信息数组
		cin >> G.Vex[i];
	}
	for(int i = 0 ; i < G.vexnum ; i++){ //初始化邻接矩阵的所有值为0,如果是网,则初始化其邻接矩阵为无穷大
		for(int j = 0; j < G.vexnum ; j ++){
			G.Edge[i][j] = 0;
		}
	}
	cout << "请输入每条边依附的两个节点:" << endl;
	while(G.edgenum --){
		cin >> u >> v;
		i = locatevex(G , u);  //查找节点u 的存储下标
		j = locatevex(G , v);  //查找节点v 的存储下标
		if(i != -1 && j != -1){
			G.Edge[i][j] = G.Edge[j][i] = 1;  //将邻接矩阵设置为1
		}
	}
}

【邻接矩阵的优缺点】

优点:

  • 快速判断在两节点之间是否有边。在图中,Edge[i ][j ]=1,表示有边;Edge[i ][j ]=0,表示无边。在网中,Edge[i ][j ]=∞,表示无边,否则表示有边。时间复杂度为O (1)。
  • 方便计算各节点的度。在无向图中,邻接矩阵第i 行元素之和就是节点i 的度;在有向图中,第i 行元素之和就是节点i 的出度,第i列元素之和就是节点i 的入度。时间复杂度为O (n )。

缺点:

  • 不便于增删节点。增删节点时,需要改变邻接矩阵的大小,效率较低。
  • 不便于访问所有邻接点。访问第i 个节点的所有邻接点时,需要访问第i 行的所有元素,时间复杂度为O (n )。访问所有节点的邻接点,时间复杂度为O (n^2 )。
  • 空间复杂度高,为O (n^2 )。

在实际应用中,如果在一个程序中只用到一个图,就可以用一个二维数组表示邻接矩阵,直接输入节点的下标,省去节点信息查询步骤。有时如果图无变化,则为了方便,可以省去输入操作,直接在程序头部定义邻接矩阵。
例如,可以直接定义图的邻接矩阵如下:文章来源地址https://www.toymoban.com/news/detail-458686.html

int M[m][n] = {{0,1,0,1} , {1,0,1,1} , {0,1,0,1} , {1,1,1,0}};

到了这里,关于图的存储 —— 邻接矩阵的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 图的存储 —— 邻接矩阵

    图的结构比较复杂,任何两个节点之间都可能有关系。 图的存储分为顺序存储和链式存储。 顺序存储包括邻接矩阵和边集数组, 链式存储包括邻接表、链式前向星、十字链表和邻接多重表。 图的存储 —— 邻接矩阵 邻接矩阵通常采用一个一维数组存储图中节点的信息,采用

    2024年02月06日
    浏览(38)
  • 图的邻接矩阵存储及遍历操作

    任务描述 本关任务:要求从文件输入顶点和边数据,包括顶点信息、边、权值等,编写程序实现以下功能。 1)构造无向网G的邻接矩阵和顶点集,即图的存储结构为邻接矩阵。 2)输出无向网G的各顶点和邻接矩阵。 3)输出无向网G中顶点H的所有邻接顶点。 测试说明 平台会对

    2024年02月06日
    浏览(40)
  • 图的存储--邻接矩阵/边集数组/邻接表/链式邻接表/链式前向星

    使用二维数组w[u][v]存储点u到点v的边的权值。 一般应用在点数不多的稠密图 时间复杂度:O(n 2 ) 空间复杂度:O(n 2 ) 边集数组e[i]存储第i条边的「起点、终点、边权」。在kruskal算法中,将边按边权排序,直接存边。 时间复杂度:O(nm) 空间复杂度:O(m) 出边数组e[u][i]存储u的所

    2024年02月02日
    浏览(36)
  • 6-1 邻接矩阵存储图的深度优先遍历

    分数 20 作者 DS课程组 单位 浙江大学 试实现邻接矩阵存储图的深度优先遍历。 其中MGraph是邻接矩阵存储的图,定义如下: 函数DFS应从第V个顶点出发递归地深度优先遍历图Graph,遍历时用裁判定义的函数Visit访问每个顶点。当访问邻接点时,要求按序号递增的顺序。题目保证

    2024年02月05日
    浏览(39)
  • 图的存储结构——邻接表

    回忆邻接矩阵的顺序存储结构,其 内存空间预先分配 , 容易导致空间的溢出或者浪费 。为了使 增减结点方便,提高空间利用效率 ,引入链式存储法——邻接表。 邻接表的组成分为 表头结点表 与 边表 ,如下图所示: 由图可见,每一个边表(单链表)的表头结点存放在表

    2024年02月03日
    浏览(28)
  • 【数据结构】邻接矩阵和邻接图的遍历

    本篇文章开始学习数据结构的图的相关知识,涉及的基本概念还是很多的。 本文的行文思路: 学习图的基本概念 学习图的存储结构——本文主要介绍邻接矩阵和邻接表 对每种结构进行深度优先遍历和广度优先遍历 话不多说,狠活献上 等等,先别急,正式学习之前先认识几个

    2024年02月04日
    浏览(50)
  • 数据结构--图的存储邻接表法

    邻接矩阵: 数组实现的顺序存储,空间复杂度高,不适合存储稀疏图 邻接表: 顺序+链式存储 无向图: 边结点的数量是 2|E|, 整体空间复杂度为 O(|V| + 2|E|) 有向图: 边结点的数量是 |E|, 整体空间复杂度为 O(|V| + |E|) 图的邻接表表示方式并不唯一 color{red}图的邻接表表示方

    2024年02月16日
    浏览(44)
  • (完全解决)如何输入一个图的邻接矩阵(每两个点的亲密度矩阵affinity),然后使用sklearn进行谱聚类

    背景 网上倒是有一些关于使用sklearn进行谱聚类的教程,但是这些教程的输入都是一些点的集合,然后根据谱聚类的原理,其会每两个点计算一次亲密度(可以认为两个点距离越大,亲密度越小),假设一共有N个点,那么就是 N*N 个亲密度要计算,这特别像什么?图里面的邻

    2024年02月07日
    浏览(43)
  • 【数据结构】图的创建(邻接矩阵,邻接表)以及深度广度遍历(BFS,DFS)

    图是由顶点集合及顶点间的关系组成的一种数据结构:G = (V, E),其中: 顶点集合V = {x|x属于某个数据对象集}是有穷非空集合; E = {(x,y)|x,y属于V}或者E = {x, y|x,y属于V Path(x, y)}是顶点间关系的有穷集合,也叫 做边的集合。 完全图:在有n个顶点的无向图中,若有n * (n-1)/2条边,

    2024年02月04日
    浏览(42)
  • 数据结构--5.1图的存储结构(十字链表、邻接多重表、边集数组)

    目录 一、十字链表(Orthogonal List) 二、邻接多重表 三、边集数组 四、深度优先遍历   重新定义顶点表结点结构:  data firstIn firstOut 重新定义边表结构结点: tailVex headVex headLink tailLink        十字链表的好处就是因为把邻接表和逆邻接表整合在了一起,这样既容易找到Vi为

    2024年02月10日
    浏览(42)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包