摘要:老孫到此一游對于深度遍歷,是將圖按一個固定方向,縱向的結果,所以是一個遞歸的結構。老孫到此一游對于廣度遍歷,是將圖按一個固定方向,橫向的結果,所以是一個鏈式進出的關系,這里是用隊列,在中做隊列這種先進先出比較簡單。
前言
今天晚上無意翻到一個圖的文章,查了一下感覺網上實現和其他都好復雜,所以自己按理解搞了一下,不知道是我實現是不是錯了...感覺還好~進入正題,先還是來點理論知識,不過大多是自己的想法,不一定都對,可以糾正。內容來源來自《JavaScript 數據結構和算法》。
圖圖是一種數學模型,和數學掛勾一般都會比較復雜,所以形象的理解成最簡單的模型,點-線 模型。其實最簡單的是 1 個點的模型,涉及 2 個點還好,3 個點過后模型就會作出相應的改變。
這里用簡單的語言來說圖中的二元關系,不過還是先假設一點數學符號:
G => 表示所有的頂點集合
V => 表示頂點
E => 表示邊,抽象意義上是無向邊
那么用數學來表示就是:G=
其實根本不用理解數學的模型,我這里理解是只需要知道這是一個點-線模型就可以了。
如何表示圖呢?這里有兩種表示方法:表和矩陣,其間都是鄰接關系
這里我有一個測試圖,在網上弄的,雖然是無向圖,其實在我們代碼中,肯定是有向的,是入口的問題:
圖的結構確定過后,就可以做出表的結構了,這里我沒有用方向,因為我理解的圖是一個不能簡單表示的,理解成坐標系更好理解一點。分為:x, y 軸的方式。其中,x0 表示開始,后面表示相鄰的點,按順時針排列(不一定按這個順序)。
代碼中的圖在代碼中表示,沒有圖形那么直觀,所以需要映射成代碼模型,這里簡單實現一下,但是不具備很多功能。
假設:
class G => 一個圖的類,包括圖的定義和常用遍歷方法
this.V => 表示點集合的個數,但是這里我舍棄了 0 的位置
this.T => 我按數據庫表的方式理解命名的,關系的集合
this.E => 邊的個數
this.visited => 訪問過的 bool 集合,其實就是標記
this.defined => 工具小函數,是否定義過,與圖無關
所以最后有關的符號有:
G、V、T、E
是不是感覺一下子變簡單了,不過程序的抽象有一個上層,那就是類。
然后我這里按計算機的方式,定義了輸入、輸出函數:input、output
class G { constructor(V) { this.V = V; this.T = []; this.E = 0; this.visited = []; for (let v = 0; v < this.V; ++v) { this.T[v] = []; this.T[v].push(-1); } this.defined = s => s !== void 0; } input(v, w) { this.T[v].push(w); this.T[w].push(v); this.E++; return this; } output() { console.table(this.T); } }
然后能夠看出,其實邊是由點的連接組成的,剛好符合數學的定義,并且與相鄰有相關性。
那么,實現了結構,還應該有其他作用,那么接下來看一下遍歷算法:深度遍歷(DFS) 和 廣度遍歷(BFS)。準確來說應該是優先采用什么策略的遍歷方式。其實我這里的實現感覺...不好,和樹關系大了點,不過樹的大集合就能夠上升到圖。
dfs(v) { this.visited[v] = true; if (this.defined( this.T[v] )) { console.log("老孫到此一游:" + v); } this.T[v].forEach(t => { if (t !== -1 && !this.visited[t]) { this.dfs(t); } }); }
對于深度遍歷,是將圖按一個固定方向,縱向的結果,所以是一個遞歸的結構。
bfs(node) { this.visited[node] = true; var queue = []; queue.push(node); while(queue.length > 0) { var v = queue.shift(); if(this.defined( this.T[v] )) { console.log("老孫到此一游:" + v); } this.T[v].forEach(t => { if(t !== -1 && !this.visited[t]) { this.visited[t] = true; queue.push(t); } }); } }
對于廣度遍歷,是將圖按一個固定方向,橫向的結果,所以是一個鏈式進出的關系,這里是用隊列,在 JS 中做隊列這種先進先出比較簡單。
// 測試代碼 var v = [1, 2, 3, 4, 5]; let g = new G( v.length + 1 ); g.input(1, 2).input(1, 5) .input(2, 4).input(2, 5).input(2, 3) .input(3, 4) .input(4, 5) .output(); g.dfs(1); console.log("------------"); // 讓它失憶一下 g.visited = []; g.bfs(1);
……-_-# 簡單玩一下,睡覺了 zZZ
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/91260.html
摘要:如果同級父元素不是層疊上下文元素就不需要看父元素的眼色了文章到這里就結束了,希望看完這篇文章的同學可以徹底理解。 今天寫代碼用antd-mobile的checkbox時候,想在內容文本后面添加一個icon,并且需要對這個icon綁定事件,發現綁定之后怎么也點不中,調試發現原來被層層嵌套的dom元素蓋住了,肯定是z-index在作祟。可是按照我之前對z-index的了解(自信滿滿)卻怎么...
摘要:如果同級父元素不是層疊上下文元素就不需要看父元素的眼色了文章到這里就結束了,希望看完這篇文章的同學可以徹底理解。 今天寫代碼用antd-mobile的checkbox時候,想在內容文本后面添加一個icon,并且需要對這個icon綁定事件,發現綁定之后怎么也點不中,調試發現原來被層層嵌套的dom元素蓋住了,肯定是z-index在作祟。可是按照我之前對z-index的了解(自信滿滿)卻怎么...
摘要:問題在說閉包,一定會牽涉到作用域。這也是閉包的屬性的,能夠記錄下內部函數引用外部的值。因為都是全局變量,所以循環也就是不斷值覆蓋,閉包并不會記錄在循環時的值,只會記錄閉包變量。閉包時記錄的除了閉包變量還有塊級作用域變量最后來看看這個輸出什么 js 是非常靈活的語言,寫起來真是* 不過現在有了typescript,寫起來舒服多了。 問題 在說js閉包,一定會牽涉到作用域。而一般在區別 v...
摘要:前言微信小程序的開發,我應該算是趕上了第一波,所以,自然是一路踩坑而來。注以下標題是按照微信開發工具上的選項進行劃分的。不過,除此之外,它還會產生另外一個副作用,就是可能連小程序本身上的請求都請求不了了。 -- KChris 2017.3.16 (=^.^=) 前言微信小程序的開發,我應該算是趕上了第一波,所以,自然是一路踩坑而來 =。=一月九日,小程序正式上線,早早地就到公司開始改b...
閱讀 1937·2021-11-24 09:39
閱讀 3522·2021-09-28 09:36
閱讀 3291·2021-09-06 15:10
閱讀 3446·2019-08-30 15:44
閱讀 1159·2019-08-30 15:43
閱讀 1802·2019-08-30 14:20
閱讀 2719·2019-08-30 12:51
閱讀 2038·2019-08-30 11:04