014 图的数据存储结构

图的逻辑结构定义

因为图的结构比较复杂,任意两个点之间都可能存在联系, 因此无法以数据元素在内存中的物理位置来表示元素之间的关系, 也就是说, 图不可能用简单的顺序结构来表示。 下面是图的物种不同的存储结构

1 邻接矩阵

顶点不分大小,主次, 所以用一个一位数组来存储顶点是不错的选择。
而边或弧是由于顶点和顶点之间的关系确定, 一维数组搞不定, 所以考虑用一个二维数组来存储, 于是我们的邻接矩阵的方案就诞生了。

1.1 无向图例

举个例子: 下图就是一个无向图,


image.png

顶点数组为 vertex[4]={v0,v1,v2,v3} , 边数组 arc[4][4],
对于矩阵的主对角线的值, 即 arc[0][0], arc[1][1], arc[2][2], arc[3][3], 全为0, 是因为不存在顶点到自身的边。

根据这个矩阵, 我们可以很容易的知道图中的信息,
1 判断任意两顶点是否有边无边就很容易了
2 我们要知道某个顶点的度, 其实就是这个顶点 Vi 在邻接矩阵中第 i 行(或者第i列)的元素之和, 比如顶点 V1 的度就是 1 + 0 + 1 + 0 = 2
3 求顶点Vi 的所有的邻接点, 就是将矩阵中第i 行元素扫描一遍, arc[i][j] 为 1 就是邻接点

1.2 有向图例

有向图讲究出度和入度, 顶点V1 的入度为1, 整好是 V1 列各数之和, 顶点V1 的出度为2, 即为第V1 行的各数之和

判断顶点是否存在弧, 只需要查找 arc[i][j] 是否为1 即可。若要求Vi 的所有邻接点, 那就把矩阵第 i 行元素都扫描一遍, 查找 arc[i][j] 为1 的顶点

image.png
1.3 网

网: 每条边上带有权的图叫做网, 那么这些全值需要保存下来, 如何处理这个矩阵需求呢 ?
把 arc 矩阵中的 1 替换成 权重即可

如下图就是一个有向图网, 右图是它的邻接矩阵。
大于0 的数字是权重,

image.png

2 邻接表

邻接矩阵的缺点: 当变数相对于顶点较少的图, 用邻接矩阵会造成较大的空间浪费, 如下图, 邻接矩阵中除了 arc[1][0] 有权值外, 没有其他弧, 这些存储空间都被浪费掉了。

image.png

我们可以用数组和线性表结合起来,用于存储图, 这种方法就叫做邻接表。

邻接表的处理办法:
1 图中顶点用一个一位数组进行存储, 对于顶点数组中, 每个数据元素还需要存储指向第一个邻接点的指针, 以便于查找该顶点的边信息
2 图中每个顶点Vi 的所有邻接点构成一个线性表, 由于邻接点的个数不定, 所以用单链表存储,, 无向图称为顶点Vi的边表, 有向图则称为顶点 Vi 作为弧尾的出边表

2.1 无向图邻接表

顶点表的各个结点, 由data 和firstedge两个域表示 , data是数据域, 存储顶点的信息, firstedge是指针域, 指向边表的第一个结点, 即此结点的第一个邻接点 。 边表结点又 adjvex 和 next 两个域组成, adjvex是邻接域, 存储某顶点的邻接点在顶点表中的下标, next则存储指向边表下一个结点的指针。 比如V1 顶点和V0 V2 互为邻接点, 则再V1 的边表中, adjvex分别为V0 的0 和 V2 的 2

image.png
2.2 有向图邻接表

有向图由于有方向, 所以我们以顶点为尾弧来存储边表, 这样容易得出每个顶点的出度, 但也有时为了便于确定顶点的入度或以顶点为弧头的弧, 我们可以建立一个有向图的逆序邻接表,

image.png

逆序邻接表


image.png

通过邻接表, 我们很容易判断出某个顶点的入度或出度为多少, 判断两顶点是否存在弧也很容易

对于带权值的网图, 可以在边表结点定义中, 再增加一个weight的数据域, 存储权值信息即可


image.png

3 十字链表

对于有向图来说, 邻接表是有缺陷的, 关心了入度问题, 想要了解出度就需要遍历整个图 , 关心了出度问题, 想要了解入度就需要遍历整个图。 我们可以通过十字表来解决这个问题

我们重新定义顶点表结构入下所示, 其中firstin 表示入边表头指针, firstout表示出边表头指针, 指向该顶点的出边表中的第一个结点,

image.png

重新定义的边表结点结构如下所示:
tailvex 表示弧七点在顶点表的下标, headvex是指弧终点在顶点表中的下标, headlink是指入边表指针域, 指向终点相同的下一条边, taillink是指边表指针域, 指向七点相同的下一条边, 如果是网, 还可以再增加一个weight来存储权值。

image.png

顶点依然存入一个一位数组, 以顶点V0 来说, firstout指向的是出边表中第一个结点V3, 所以V0边表结点的 headvex=3, 而tailvex其实就是当前顶点V0的下标0, 用于V0只有一个出边顶点, 所以 headlink 和taillink都是空

image.png

0 3 表示从 V0 到V3 的边, 所以顶点0的 firstout 指向了这条边的指针,
而 V3 的firstInt , 也指向了这条边()的指针(在图中就是虚线⑤)

其他顶点的firstin firstout均可这么理解

image.png

,

接下来解释虚线箭头的含义, 它其实就是此图的逆序邻接表的表示, 对于V0lduo , 它有两个顶点,
V1 V2的入边, 因此 V0 的firstin 指向顶点V1 的边表结点中 headvex 为0的结点, 接着由入边 结点的headlink指向下一个入边顶点V2的边表结点中 headvex为1的结点, 如图中的③, 顶点V2 和V3 也是同样有一个入边顶点, 如图中④和⑤

4 邻接多重表

重新定义的边表结点结构如下所示:

image.png

其中ivex 和jvex是与某条边依附的两个顶点在 顶点表 中的下标,
ilink 指向依附顶点ivex的下一条边,jlink指向依附顶点jvex的下一条边。

如下图,总共四个顶点五条边, 因为是无向图, 所以ivex 是0, jvex是1 还是反过来都可以, 不过为了绘图方便, 都将ivex的值设置得和旁边顶点的下标一致


image.png

接下来,我们开始把他们链接起来,
1.首先连线的 ①②③④ 就是将顶点的firstedge指向一条边, 顶点下标要与ivex的值相同
2.由于顶点V0 的(V0,V1)的边有邻边(V0,V2), (V0,V3);
所以 ⑤指向的就是 (V0,V3)这个邻边, 它依附于顶点 V0, 对应的是ivex=0 时候的ilink
⑥指向的是 (V0,V3)这个边的邻边, 只是这时候 V0 在 jvex的位置
⑦指的就是(V1,V0) 这条边
⑧指的就是(V1,V2) 这条边
⑨指的就是(V0,V2) 这条边

image.png

5 边集数组

边集数组由两个一维数组构成, 一个是存储顶点的信息, 另一个存储边的信息, 这个边数组每个数据元素由一条边的起点下标 begin, 终点下标 end 和权 weight组成。很明显, 边集数组关注的是边的集合, 在边集数组中要查询一个顶点的度需要扫描整个边数组, 效率并不高。

image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,125评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,293评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,054评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,077评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,096评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,062评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,988评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,817评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,266评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,486评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,646评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,375评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,974评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,621评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,642评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,538评论 2 352

推荐阅读更多精彩内容

  • 一. 有向图/无向图 共有:邻接表,邻接矩阵 有向图独有:十字链表,边集数组 无向图独有:邻接多重表 1.邻接矩...
    良宵Zzz阅读 1,112评论 0 0
  • 1. 图的定义和基本术语 线性结构中,元素仅有线性关系,每个元素只有一个直接前驱和直接后继;树形结构中,数据元素(...
    yinxmm阅读 5,450评论 0 3
  • 在数据结构中图算是个较为难理解的结构形式了。 大致我们可以分为两个大类:1、通过数组实现2、通过链表实现 而链表的...
    飞扬code阅读 6,032评论 0 3
  • 图(Graph)是数据结构中最复杂的一种结构,线性表描述的是一对一关系,树描述的是一对多关系,而图描述的是多对多关...
    大大纸飞机阅读 1,826评论 0 3
  • 内容整理于鱼c工作室教程 1. 图的基本概念 1.1 图的概念 图(Graph)是由顶点的有穷非空集合和顶点之间边...
    阿阿阿阿毛阅读 3,179评论 0 2