数据结构与算法JavaScript描述(9) —— 图(Graph)

1. 图的定义

图由的集合及顶点的集合组成。
边由顶点对(v1, v2)定义,v1和v2分别是图中的两个顶点。
顶点也有权重,也称为成本。
图中的一系列顶点构成路径,路径中所有的顶点都由边连接。
路径的长度用路径中第一个顶点最后一个顶点之间边的数量表示。

如果一个图的顶点对是有序的,则可以称之为有向图。有向图表明了顶点的流向。


有向图.png

如果图是无序的,则称之为无序图,或无向图:


无序图.png

本篇主要讨论无序图。

表示图的边的方法称为邻接表或者邻接数组
这种方法将边存储为由顶点的相邻顶点列表构成的数组,并以此顶点作为索引。

邻接表.png

2. 图算法 —— 搜索图

(1)深度优先搜索
深度优先搜索包括从一条路径的起始顶点开始追溯,直到到达最后一个顶点,然后回溯,继续追溯下一条路径,直到到达最后的顶点,如此往复,直到没有路径为止。这不是在搜索特定的路径,而是通过搜索来查看在图中有哪些路径可以选择。

深度优先搜索.png

(2)广度优先搜索
广度优先搜索从第一个顶点开始,尝试尽可能靠近它的顶点。本质上,这种搜索在图上是逐层移动的,首先检查最靠近第一个顶点的层,再逐渐向下移动到离起始顶点最远的层。
广度优先算法使用了抽象队列来对已访问过的顶点进行排序。原理如下:

a. 查找与当前顶点相邻的未访问顶点,将其添加到已访问顶点列表及队列中;
b. 从图中取出下一个顶点v,添加到已访问的顶点列表
c. 将所有与v相邻的未访问顶点添加到队列
广度优先搜索.png

3. 实现一个图类

class Graph {

    constructor(v = 0, vertexList = []) {
        this.vertices = v // 顶点数
        this.edges = 0  // 边数
        this.adj = []       // 邻接数组
        this.marked = []    // 存储已访问的节点
        this.edgeTo = []    // 保存一个顶点到下一个顶点的所有边
        this.vertexList = vertexList    // 将各个顶点关联到一个符号
        // 为每个顶点初始化一个数组用来存储与他相连的节点
        for (let i = 0; i < this.vertices; i++) {
            this.adj[i] = []
        }
        // 初始化所有顶点为未访问过
        for (let i = 0; i < this.vertices; i++) {
            this.marked[i] = false
        }
    }

    // 添加一条边
    addEdge(v, w) {
        this.adj[v].push(w)
        this.adj[w].push(v)
        this.edges++
    }

    // 深度优先搜索
    // 访问一个没有被访问过的顶点,将它标记为已访问,再递归地去访问在初始顶点的邻接表中其他没有被访问顶点
    dfs(v = 0) {
    this.marked[v] = true
    if (this.adj[v] !== undefined) {
        console.log(`Visited vertex: ${v}`)
        this.adj[v].forEach(item => {
            if (!this.marked[item]) {
                this.dfs(item)
            }
        })
    }
    }

    // 广度优先搜索
    // (1) 查找与当前顶点相邻的未访问顶点,将其添加到已访问顶点列表及队列中
    // (2) 从图中取出下一个顶点v,添加到已访问的顶点列表中
    // (3) 将所有与v相邻的未访问顶点添加到队列中
    bfs(s = 0) {
        this.resetMarked()
        const queue = []
        this.marked[s] = true
        queue.push(s) // 添加到队尾
        while(queue.length > 0) {
            const v = queue.shift() // 从队首移除
            if (this.adj[v] !== undefined) {
                console.log(`Visited vertex: ${v}`)
                // 遍历访问与v的邻接表中其他没有被访问的顶点
                this.adj[v].forEach(item => {
                    if (!this.marked[item]) {
                        // 保存 item => v 的边
                        this.edgeTo[item] = v
                        // 将 item 标记为已访问
                        this.marked[item] = true
                        queue.push(item)
                    }
                })
            }
        }
        this.resetMarked()
    }

    // 存储与指定顶点有共同边的所有顶点
    pathTo(v) {
        this.dfs()
        let source = 0
        if (!this.marked[v]) return undefined
        const path = []
        let i = v
        while  (i !== source) {
            if (this.edgeTo[i] !== undefined) {
                path.push(i)
                i = this.edgeTo[i]
            } else {
                i = source
            }
        }
        path.push(source)
        this.resetMarked()
        return path.reverse()
    }

    // 拓扑排序
    // 设置排序进程并调用一个辅助函数 topSortHelper()
    // 然后显示排序好的顶点列表
    topSort() {
        const stack = []
        const visited = []
        for (let i = 0; i < this.vertices; i++) {
            visited[i] = false
        }
        for (let i = 0; i < this.vertices; i++) {
            if (!visited[i]) {
                this.topSortHelper(i, visited, stack)
            }
        }
        for (let i = stack.length - 1; i >= 0; i--) {
            if (stack[i] !== undefined) {
                console.log(this.vertexList[stack[i]])
            }
        }
    }

    // 拓扑排序辅助函数
    // 将当前顶点标记为已访问,然后递归地访问当前顶点邻接表中每个相邻顶点,标记这些顶点为已访问。
    // 最后将当前顶点压入栈
    topSortHelper(v, visited, stack) {
        visited[v] = true
        this.adj[v].forEach(item => {
            if (!visited[item]) {
                this.topSortHelper(item, visited, stack)
            }
        })
        stack.push(v)
    }

    showGraph() {
        const visited = []
        for (let i = 0; i < this.vertices; i++) {
            console.log(' ')
            visited.push(this.vertexList[i])
            for (let j = 0; j < this.vertices; j++) {
                if (this.adj[i][j] !== undefined && visited.indexOf(this.vertexList[j] === -1)) {
                    console.log(`${this.vertexList[i]} -> ${this.adj[i][j]}`)
                }
                visited.pop()
            }
        }
    }

    // 重置所有顶点为未访问过
    resetMarked() {
        for (let i = 0; i < this.vertices; i++) {
            this.marked[i] = false
        }
    }
}



// test
const lessons = ['CS1', 'CS2', 'Data Structures',
                     'Assembly Language', 'Operating Systems',
                     'Algorithms']
const g = new Graph(6, lessons)
g.addEdge(1, 2)
g.addEdge(2, 5)
g.addEdge(1, 3)
g.addEdge(1, 4)
g.addEdge(0, 1)

console.log('======= showGraph =======')
g.showGraph()

console.log('======= 深度优先 =======')
g.dfs()
// Visited vertex: 0
// Visited vertex: 1
// Visited vertex: 2
// Visited vertex: 5
// Visited vertex: 3
// Visited vertex: 4

console.log('======= 广度优先 =======')
g.bfs() 
// Visited vertex: 0
// Visited vertex: 1
// Visited vertex: 2
// Visited vertex: 3
// Visited vertex: 4
// Visited vertex: 5

console.log('拓扑排序 =======')
g.topSort() 
// CS1
// CS2 
// Operating Systems
// Assembly Language
// Data Structures
// Algorithms 

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

推荐阅读更多精彩内容