8-图(第11章)
8.1 图的定义
图是一种非线性结构,由一系列顶点及其连接顶点的边组成。比如A和B、A和D是相邻的,而A和E不是相邻的。一个顶点相邻顶点的数量叫作度,比如A的度为3、D的度为4。路径指相邻顶点的一个连续序列,如ABEI、ACDG;简单路径指不包含重复顶点的路径(除环外),如ADG;环指首尾顶点相同的路径,如ADCA,环也属于简单路径。如果图中每两个顶点之间都有路径相连,则称该图是连通的。

如果图中的边具有方向,则成为有向图。如果图中的边是双向的,则称图是强连通的。比如下图中的C和D是强连通的

8.2 图的表示
8.2.1 顶点
创建图类的第一步就是要创建一个 Vertex 类来保存顶点和边,Vertex 类有两个数据成员:一个用于标识顶点,另一个是表明这个顶点是否被访问过的布尔值 。实现这个类的程序如下:
1 2 3 4 5 6
| class Vertex { constructor(label, color) { this.label = label this.color = color } }
|
(原书上关于顶点的实现跟我上面的程序还不一样。其实顶点不需要用程序实现的,接下来也用不到)
8.2.2 边
图的实际信息都保存在边上面,因为它们描述了图的结构。表示图的方法有好几种,我们选用最常见的邻接表法表示边。这种方法将边存储为由顶点的相邻顶点列表构成的数组,并以此顶点作为索引。使用这种方案,当我们在程序中引用一个顶点时,可以高效地访问与这个顶点相连的所有顶点的列表。比如,如果顶点 2 与顶点 0、1、 3、 4 相连,我们就可以在map键值为2的位置,存储它的相邻顶点0、1、3、4

8.2.3 图
程序如下:这个类会记录一个图表示了多少条边,并使用数组来记录顶点数。使用一个键值为数组的hash表来存储顶点的相邻顶点
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30
| class Graph { constructor() { this.vertices = [] this.adj = new Map() } addVertice(v) { this.vertices.push(v) this.adj.set(v, []) } addEdge(v, w) { this.adj.get(v).push(w) this.adj.get(w).push(v) } showGraph() { let keys = this.adj.keys() for (let k of keys) { console.log(k + '->') for (let i of this.adj.get(k)) { console.log(i + ' ') } } } }
|
测试:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
| let g = new Graph() let vertices = [1, 2, 3, 4, 5, 6, 7]
vertices.forEach(v => { g.addVertice(v) })
g.addEdge(1, 2) g.addEdge(1, 3) g.addEdge(1, 6) g.addEdge(2, 7) g.addEdge(3, 5) g.addEdge(3, 4) g.addEdge(3, 6) g.showGraph()
1-> 2 3 6 2-> 1 7 3-> 1 5 4 6 4-> 3 5-> 3 6-> 1 3 7-> 2
|
打印结果正确,说明我的程序没有问题。
8.3 图的遍历
图的遍历有两种常用的方法:深度优先搜索和广度优先搜索。
8.3.1 深度优先搜索
深度优先搜索包括从一条路径的起始顶点开始追溯,直到到达最后一个顶点,然后回溯,继续追溯下一条路径,直到到达最后的顶点,如此往复,直到没有路径为止。

深度优先搜索算法比较简单:访问一个没有访问过的顶点,将它标记为已访问,再递归地去访问在初始顶点的邻接表中其他没有访问过的顶点。
8.3.2 广度优先遍历
广度优先搜索从第一个顶点开始,尝试访问尽可能靠近它的顶点。本质上,这种搜索在图上是逐层移动的,首先检查最靠近第一个顶点的层,再逐渐向下移动到离起始顶点最远的层。

8.4 程序实现
假设我们有以下一个图,我们尝试下实现这个图,并且分别用深度遍历和广度遍历访问这个图

原书中对图的实现太粗糙了,原书中图要求每个顶点都是数字,我这边做了一下优化
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86
| class Graph { constructor() { this.vertices = [] this.adj = new Map() } addVertice(v) { this.vertices.push(v) this.adj.set(v, []) } addEdge(v, w) { this.adj.get(v).push(w) this.adj.get(w).push(v) } showGraph() { let keys = this.adj.keys() for (let k of keys) { console.log(k + '->') for (let i of this.adj.get(k)) { console.log(i + ' ') } } } initialColor() { let color = new Map() this.vertices.forEach(ele => { color.set(ele, 'white') }) return color }
dfs(callback) { let color = this.initialColor() this.vertices.forEach(node => { if (color.get(node) === 'white') { this.dfsVisit(node, color, callback) } }) } dfsVisit(node, color, callback) { let neighbors = this.adj.get(node) color.set(node, 'grey') if (callback) { callback(node) } neighbors.forEach(neighNode => { if (color.get(neighNode) === 'white') { this.dfsVisit(neighNode, color, callback) } }) color.set(node, 'black') } bfs(callback) { let color = this.initialColor() let queue = [] if (this.vertices[0]) { queue.push(this.vertices[0]) } while (queue.length > 0) { let qNode = queue.shift() let neighbors = this.adj.get(qNode) color.set(qNode, 'grey') neighbors.forEach(ele => { if (color.get(ele) === 'white') { color.set(ele, 'grey') queue.push(ele) } }) color.set(qNode, 'black') if (callback) { callback(qNode) } } } }
|
上面的程序实现图的定义,还实现了深度遍历和广度遍历的方法。我们队程序测试下
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
| let g = new Graph() let vertices = [1, 2, 3, 4, 5, 6, 7] vertices.forEach(v => { g.addVertice(v) }) g.addEdge(1, 2) g.addEdge(1, 3) g.addEdge(1, 6) g.addEdge(2, 7) g.addEdge(3, 5) g.addEdge(3, 4) g.addEdge(3, 6)
g.dfs(v=>console.log(v))
g.bfs(v=>console.log(v))
|
测试结果是正确的,说明我们的程序实现没有问题
8.5 查找最短路径
图的最常见的操作是查找从一个顶点到另一个顶点的最短路径,就要用到广度搜索。要查找从顶点 A 到顶点 D 的最短路径,我们首先会查找从 A 到 D 是否有任何一条单边路径,接着查找两条边的路径,以此类推。这正是广度优先搜索的搜索过程,因此我们可以轻松地修改广度优先搜索算法,找出最短路径。
代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62
| bfsDistance() { let color = this.initialColor() let queue = [] let distance = new Map() let pred = new Map() if (this.vertices[0]) { queue.push(this.vertices[0]) } this.vertices.forEach(ele => { distance.set(ele, 0) pred.set(ele, null) }); while (queue.length > 0) { let qNode = queue.shift() let neighbors = this.adj.get(qNode) color.set(qNode, 'grey') neighbors.forEach(ele => { if (color.get(ele) === 'white') { color.set(ele, 'grey') queue.push(ele) distance.set(ele, distance.get(qNode) + 1) pred.set(ele, qNode) } }) color.set(qNode, 'black') } return { distance: distance, pred: pred } }
getBfsPath() { if (this.vertices.length > 0) { let shortestPath = this.bfsDistance() let fromVertex = this.vertices[0] let paths = [] for (let i = 0; i < this.vertices.length; i++) { let path = [] let toVertex = this.vertices[i] for (let v = toVertex; v !== fromVertex; v = shortestPath.pred.get(v)) { path.push(v) } path.push(fromVertex) paths.push(path) } return paths } }
getShortestPath(v) { let path = null let paths = this.getBfsPath() for (let i = 0; i < paths.length; i++) { if (paths[i][0] === v) { path = paths[i] break } } return path }
|
测试:
我们以获取顶点到第4个点的最短路径为例,测试如下
1 2
| g.getShortestPath(g.vertices[4])
|
案例获取顶点到某个点的最短路径,怎么样获取非顶点的两个点之间的最短路径呢? 其实道理很简单,你换下顶点就可以了。在图里,并没有规定那个必须是顶点。
8.6 拓扑排序
拓扑排序会对有向图的所有顶点进行排序,使有向图从前面的顶点指向后面的顶点。如下图是计算机学科的有向图模型。

这个图的拓扑排序结果将会是以下序列:
- CS1
- CS2
- 汇编语言
- 数据结构
- 操作系统
- 算法
课程3和课程4可以同时上,课程5和课程6也可以同时上,但是其他课程就要有优先级了。比如c1一定要在cs2之前上。
拓扑排序代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
| topSort() { let stack = [] let color = this.initialColor() this.vertices.forEach(ele => { if (color.get(ele) === 'white') { this.topSortVisit(ele, stack, color) } }) console.log('stack:', stack) } topSortVisit(ele, stack, color) {
color.set(ele, 'grey') stack.push(ele) let neighbors = this.adj.get(ele) neighbors.forEach(nNode => { if (color.get(nNode) === 'white') { this.topSortVisit(nNode, stack, color) } }) }
|
用已有的案例测试:
用其他案例测试下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| let g = new Graph() let vertices = ['cs1', 'cs2', '汇编语言', '数据结构', '操作系统', '算法']
vertices.forEach(v => { g.addVertice(v) })
g.addEdge('cs1', 'cs2') g.addEdge('cs2', '汇编语言') g.addEdge('cs2', '数据结构') g.addEdge('数据结构', '操作系统') g.addEdge('数据结构', '算法') g.topSort() // stack: (6) ["cs1", "cs2", "汇编语言", "数据结构", "操作系统", "算法"]
|
测试应该是没问题的。
本章节完整代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186
| class Graph { constructor() { this.vertices = [] this.adj = new Map() } addVertice(v) { this.vertices.push(v) this.adj.set(v, []) } addEdge(v, w) { this.adj.get(v).push(w) this.adj.get(w).push(v) } showGraph() { let keys = this.adj.keys() for (let k of keys) { console.log(k + '->') for (let i of this.adj.get(k)) { console.log(i + ' ') } } } initialColor() { let color = new Map() this.vertices.forEach(ele => { color.set(ele, 'white') }) return color }
dfs(callback) { let color = this.initialColor() this.vertices.forEach(node => { if (color.get(node) === 'white') { this.dfsVisit(node, color, callback) } }) } dfsVisit(node, color, callback) { let neighbors = this.adj.get(node) color.set(node, 'grey') if (callback) { callback(node) } neighbors.forEach(neighNode => { if (color.get(neighNode) === 'white') { this.dfsVisit(neighNode, color, callback) } }) color.set(node, 'black') } bfs(callback) { let color = this.initialColor() let queue = [] if (this.vertices[0]) { queue.push(this.vertices[0]) } while (queue.length > 0) { let qNode = queue.shift() let neighbors = this.adj.get(qNode) color.set(qNode, 'grey') neighbors.forEach(ele => { if (color.get(ele) === 'white') { color.set(ele, 'grey') queue.push(ele) } }) color.set(qNode, 'black') if (callback) { callback(qNode) } } } bfsDistance() { let color = this.initialColor() let queue = [] let distance = new Map() let pred = new Map() if (this.vertices[0]) { queue.push(this.vertices[0]) } this.vertices.forEach(ele => { distance.set(ele, 0) pred.set(ele, null) }); while (queue.length > 0) { let qNode = queue.shift() let neighbors = this.adj.get(qNode) color.set(qNode, 'grey') neighbors.forEach(ele => { if (color.get(ele) === 'white') { color.set(ele, 'grey') queue.push(ele) distance.set(ele, distance.get(qNode) + 1) pred.set(ele, qNode) } }) color.set(qNode, 'black') } return { distance: distance, pred: pred } } getBfsPath() { if (this.vertices.length > 0) { let shortestPath = this.bfsDistance() let fromVertex = this.vertices[0] let paths = [] for (let i = 0; i < this.vertices.length; i++) { let path = [] let toVertex = this.vertices[i] for (let v = toVertex; v !== fromVertex; v = shortestPath.pred.get(v)) { path.push(v) } path.push(fromVertex) paths.push(path) } return paths } } getShortestPath(v) { let path = null let paths = this.getBfsPath() for (let i = 0; i < paths.length; i++) { if (paths[i][0] === v) { path = paths[i] break } } return path } topSort() { let stack = [] let color = this.initialColor() this.vertices.forEach(ele => { if (color.get(ele) === 'white') { this.topSortVisit(ele, stack, color) } }) console.log('stack:', stack) } topSortVisit(ele, stack, color) {
color.set(ele, 'grey') stack.push(ele) let neighbors = this.adj.get(ele) neighbors.forEach(nNode => { if (color.get(nNode) === 'white') { this.topSortVisit(nNode, stack, color) } }) } }
let g = new Graph() let vertices = [1, 2, 3, 4, 5, 6, 7]
vertices.forEach(v => { g.addVertice(v) })
g.addEdge(1, 2) g.addEdge(1, 3) g.addEdge(1, 6) g.addEdge(2, 7) g.addEdge(3, 5) g.addEdge(3, 4) g.addEdge(3, 6) g.getBfsPath()
|
个人评价:关于图的知识常用的大概就这么多,书上有关图的代码写的真是一团糟,很多要自己查资料实现。这作者真是太不负责任了