版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請(qǐng)進(jìn)行舉報(bào)或認(rèn)領(lǐng)
文檔簡(jiǎn)介
1、圖的基本概念圖的存儲(chǔ)表示圖的遍歷與連通性 最小生成樹最短路徑 活動(dòng)網(wǎng)絡(luò),第八章 圖,圖的基本概念,圖定義 圖是由頂點(diǎn)集合(vertex)及頂點(diǎn)間的關(guān)系集合組成的一種數(shù)據(jù)結(jié)構(gòu): Graph=( V, E ) 其中 V = { x | x ? 某個(gè)數(shù)據(jù)對(duì)象} 是頂點(diǎn)的有窮非空集合; E = {(x, y) | x, y ? V } 或 E
2、 = { | x, y ? V && Path (x, y)} 是頂點(diǎn)之間關(guān)系的有窮集合,也叫做邊(edge)集合。Path (x, y)表示從 x 到 y 的一條單向通路, 它是有方向的。,,,,,,,,,,,,,,,,,,,有向圖與無(wú)向圖 在有向圖中,頂點(diǎn)對(duì) 是有序的。在無(wú)向圖中,頂點(diǎn)對(duì)(x, y)是無(wú)序的。完全圖 若有 n 個(gè)頂點(diǎn)的無(wú)向圖有 n(n-1)/2 條邊, 則此圖為完全無(wú)向圖。有 n
3、個(gè)頂點(diǎn)的有向圖有n(n-1) 條邊, 則此圖為完全有向圖。,,,,,,,,,,,,,,,,,,,,,,,,,,0,0,0,0,1,1,1,1,2,2,2,2,6,5,4,3,3,,,,鄰接頂點(diǎn) 如果 (u, v) 是 E(G) 中的一條邊,則稱 u 與 v 互為鄰接頂點(diǎn)。子圖 設(shè)有兩個(gè)圖 G=(V, E) 和 G‘=(V’, E‘)。若 V’? V 且 E‘?E, 則稱 圖G’ 是 圖G 的子圖。權(quán) 某些圖的邊具有與
4、它相關(guān)的數(shù), 稱之為權(quán)。這種帶權(quán)圖叫做網(wǎng)絡(luò)。,,,,,,,,,0,1,2,3,,子圖,,,,,,0,1,3,,,,,,,0,1,2,3,,,,,0,2,3,頂點(diǎn)的度 一個(gè)頂點(diǎn)v的度是與它相關(guān)聯(lián)的邊的條數(shù)。記作TD(v)。在有向圖中, 頂點(diǎn)的度等于該頂點(diǎn)的入度與出度之和。頂點(diǎn) v 的入度是以 v 為終點(diǎn)的有向邊的條數(shù), 記作 ID(v); 頂點(diǎn) v 的出度是以 v 為始點(diǎn)的有向邊的條數(shù), 記作 OD(v)。路徑 在圖 G=(V
5、, E) 中, 若從頂點(diǎn) vi 出發(fā), 沿一些邊經(jīng)過一些頂點(diǎn) vp1, vp2, …, vpm,到達(dá)頂點(diǎn)vj。則稱頂點(diǎn)序列 (vi vp1 vp2 ... vpm vj) 為從頂點(diǎn)vi 到頂點(diǎn) vj 的路徑。它經(jīng)過的邊(vi, vp1)、(vp1, vp2)、...、(vpm, vj) 應(yīng)是屬于E的邊。,路徑長(zhǎng)度 非帶權(quán)圖的路徑長(zhǎng)度是指此路徑上邊的條數(shù)。帶權(quán)圖的路徑長(zhǎng)度是指路徑上各邊的權(quán)之和。簡(jiǎn)單路徑 若路徑上各頂點(diǎn) v1,
6、v2,...,vm 均不 互相重復(fù), 則稱這樣的路徑為簡(jiǎn)單路徑?;芈?若路徑上第一個(gè)頂點(diǎn) v1 與最后一個(gè)頂點(diǎn)vm 重合, 則稱這樣的路徑為回路或環(huán)。,,,,,,,,,,,0,1,2,3,,,,,,,,,,,0,1,2,3,,,,,,,,,,,0,1,2,3,,,,,,,,,,,,連通圖與連通分量 在無(wú)向圖中, 若從頂點(diǎn)v1到頂點(diǎn)v2有路徑, 則稱頂點(diǎn)v1與v2是連通的。如果圖中任意一對(duì)頂點(diǎn)都是連通的, 則稱此圖是連通圖。非
7、連通圖的極大連通子圖叫做連通分量。強(qiáng)連通圖與強(qiáng)連通分量 在有向圖中, 若對(duì)于每一對(duì)頂點(diǎn)vi和vj, 都存在一條從vi到vj和從vj到vi的路徑, 則稱此圖是強(qiáng)連通圖。非強(qiáng)連通圖的極大強(qiáng)連通子圖叫做強(qiáng)連通分量。生成樹 一個(gè)連通圖的生成樹是其極小連通子圖,在n個(gè)頂點(diǎn)的情形下,有n-1條邊。,圖的抽象數(shù)據(jù)類型class Graph {public: Graph ( ); void InsertVerte
8、x ( Type & vertex ); void InsertEdge ( int v1, int v2, int weight ); void RemoveVertex ( int v ); void RemoveEdge ( int v1, int v2 ); int IsEmpty ( ); Type GetWeight ( int v1, int v2 );
9、 int GetFirstNeighbor ( int v ); int GetNextNeighbor ( int v1, int v2 ); },,圖的存儲(chǔ)表示,在圖的鄰接矩陣表示中,有一個(gè)記錄各個(gè)頂點(diǎn)信息的頂點(diǎn)表,還有一個(gè)表示各個(gè)頂點(diǎn)之間關(guān)系的鄰接矩陣。設(shè)圖 A = (V, E)是一個(gè)有 n 個(gè)頂點(diǎn)的圖, 圖的鄰接矩陣是一個(gè)二維數(shù)組 A.edge[n][n],定義:,鄰接矩陣 (Adjacency Matr
10、ix),,,無(wú)向圖的鄰接矩陣是對(duì)稱的;有向圖的鄰接矩陣可能是不對(duì)稱的。,,,,,,,,,0,1,2,3,,0,,1,,2,,,在有向圖中, 統(tǒng)計(jì)第 i 行 1 的個(gè)數(shù)可得頂點(diǎn) i 的出度,統(tǒng)計(jì)第 j 行 1 的個(gè)數(shù)可得頂點(diǎn) j 的入度。在無(wú)向圖中, 統(tǒng)計(jì)第 i 行 (列) 1 的個(gè)數(shù)可得頂點(diǎn)i 的度。,網(wǎng)絡(luò)的鄰接矩陣,,,,,,,,,,,,,,,8,6,3,1,2,9,5,4,2,0,3,1,用鄰接矩陣表示的圖的類的定義cons
11、t int MaxEdges = 50;const int MaxVertices = 10;,template class Graph {private: Type VerticesList[MaxVertices]; //圖的頂點(diǎn)表 float Edge[MaxVertices][MaxVertices]; //圖的鄰接矩陣 int NumEdges, NumVer
12、tices; //邊數(shù)與頂點(diǎn)數(shù) int GetVertexPos ( Type vertex );public: Graph ( int sz = MaxEdges );,int GraphFull( ) const { return NumVertices == MaxVertices || NumEdges =
13、= MaxEdges; } Type GetValue ( int i ) //取頂點(diǎn)值 { return i >= 0 && i <= NumVertices ? VerticesList[i] : ‘\0’; } float GetWeight ( int u, int v ) //取邊權(quán)值 { return
14、 u != -1 && v != -1 ? Edge[u][v] : else return 0; } int NumberOfVertices ( ) //取頂點(diǎn)個(gè)數(shù) { return NumVertices; },int NumberOfEdges ( ) //取邊條數(shù) { return NumEdges; }
15、 int GetFirstNeighbor ( int v ); int GetNextNeighbor ( int u, int v ); void InsertVertex ( const Type v ); void InsertEdge ( int u, int v, float weight ); void RemoveVertex ( int v );
16、void RemoveEdge ( int u, int v );},鄰接矩陣實(shí)現(xiàn)的部分圖操作template Graph :: Graph ( int sz ) { //構(gòu)造函數(shù) NumVertices = sz; NumEdges = 0; for ( int i = 0; i < sz; i++ ) { VerticesList[i] : 0; f
17、or ( int j = 0; j < sz; j++ ) Edge[i][j] = 0;},template int Graph ::GetVertexPos ( Type vertex ) {//根據(jù)頂點(diǎn)值求它的頂點(diǎn)序號(hào) for ( int i = 0; i < NumVertices; i++ ) if ( VerticesList[i] == Vertex ) return i;
18、 return –1;},template int Graph ::GetFirstNeighbor ( const int v ) {//給出頂點(diǎn)位置為 v 的第一個(gè)鄰接頂點(diǎn)的位置 if ( v != -1 ) { for ( int col = 0; col 0 && Edge[v][col] < MaxValue ) return col; }
19、 else return -1;},template int Graph ::GetNextNeighbor ( int v, int w ) {//給出頂點(diǎn)v的某鄰接頂點(diǎn)w的下一個(gè)鄰接頂點(diǎn) int col; if ( v != -1 && w != -1 ) { for ( col = w+1; col 0 && Edge[v]
20、[col] < MaxValue ) return col; } return -1;},鄰接表 (Adjacency List),無(wú)向圖的鄰接表 同一個(gè)頂點(diǎn)發(fā)出的邊鏈接在同一個(gè)邊鏈表中,每一個(gè)鏈結(jié)點(diǎn)代表一條邊(邊結(jié)點(diǎn)), 結(jié)點(diǎn)中有另一頂點(diǎn)的下標(biāo) dest 和指針 link。,,,,,,,,A,B,C,D,data adj,,,ABCD,,,,0123,,,,,,dest link,,
21、,,,,,,,,,,,,dest link,?,?,?,?,1,3,0,2,1,0,有向圖的鄰接表和逆鄰接表,,,,A,,B,,C,,,data adj,,,ABC,,,012,,,,dest link,,,,,,,dest link,?,鄰接表 (出邊表),data adj,,,ABC,,,012,,,,dest link,,,,逆鄰接表 (入邊表),,,,1,0,2,?,?,?,?,?,0,1,1,,,,,網(wǎng)絡(luò) (帶
22、權(quán)圖) 的鄰接表,B,A,C,D,,,6,9,5,2,8,data adj,,ABCD,,,0123,,,,dest cost link,,,,?,,,,,,,,,,,,,,,?,?,?,,1 5,3 6,2 8,3 2,1 9,(出邊表),(頂點(diǎn)表),帶權(quán)圖的邊結(jié)點(diǎn)中保存該邊上的權(quán)值 cost。頂點(diǎn) i 的邊鏈表的表頭指針 adj 在頂點(diǎn)表的下標(biāo)為 i 的頂點(diǎn)記錄中,該記錄還保存了該頂點(diǎn)的其它信息。在鄰接表的
23、邊鏈表中,各個(gè)邊結(jié)點(diǎn)的鏈入順序任意,視邊結(jié)點(diǎn)輸入次序而定。設(shè)圖中有 n 個(gè)頂點(diǎn),e 條邊,則用鄰接表表示無(wú)向圖時(shí),需要 n 個(gè)頂點(diǎn)結(jié)點(diǎn),2e 個(gè)邊結(jié)點(diǎn);用鄰接表表示有向圖時(shí),若不考慮逆鄰接表,只需 n 個(gè)頂點(diǎn)結(jié)點(diǎn),e 個(gè)邊結(jié)點(diǎn)。,鄰接表表示的圖的類定義#define DefaultSize 10template class Graph;template struct Edge { //邊結(jié)點(diǎn)friend cla
24、ss Graph; int dest;//目標(biāo)頂點(diǎn)下標(biāo) float cost; //邊上的權(quán)值 Edge * link; //下一邊鏈接指針 Edge ( ) { } //構(gòu)造函數(shù) Edge ( int D, Type C ) : dest (D), cost (C),
25、link (NULL) { },int operator != ( Edge& E ) const { return dest != E.dest; } }template struct Vertex { //頂點(diǎn)friend class Graph ; Type data; //頂點(diǎn)數(shù)據(jù) Edge *adj; //邊鏈表頭指針}template class Gr
26、aph { //圖類private: Vertex * NodeTable; //頂點(diǎn)表,int NumVertices; //當(dāng)前頂點(diǎn)個(gè)數(shù) int MaxVertices; //最大頂點(diǎn)個(gè)數(shù) int NumEdges; //當(dāng)前邊數(shù) int GetVertexPos ( Type vertex );
27、public: Graph ( int sz ); ~Graph ( ); int GraphFull ( ) const { return NumVertices == MaxVertices; } Type GetValue ( int i ) { return i >= 0 && i < NumVertices ? NodeTab
28、le[i].data : 0; },int NumberOfVertices ( ) { return NumVertices; } int NumberOfEdges ( ) { return NumEdges; } float GetWeight ( int u, int v ); void InsertVertex ( Type v ); void Rem
29、oveVertex ( int v ); void InsertEdge ( int u, int v, float weight ); void RemoveEdge ( int u, int v ); int GetFirstNeighbor ( int v ); int GetNextNeighbor ( int v, int w );},鄰接表的構(gòu)造函數(shù)和析構(gòu)函數(shù)template
30、 Graph :: Graph ( int sz = DefaultSize ) : MaxVertices ( sz ) { int n, e, k, j; Type name, tail, head; float weight; NodeTable = //創(chuàng)建頂點(diǎn)表 new Vertex[MaxVertices];
31、NumVertices = 0; NumEdges = 0; cin >> n; //輸入頂點(diǎn)個(gè)數(shù),for ( int i = 0; i > name; InsertVertex ( name ); } cin >> e; //輸入邊條數(shù) for ( i = 0; i > tail >>
32、; head >> weight; k = GetVertexPos ( tail ); j = GetVertexPos ( head ); InsertEdge ( k, j, weight ); //插入邊 }},template Graph :: ~Graph ( ) { for ( int i = 0; i *p = NodeTable[i].a
33、dj; while ( p != NULL ) { //逐條邊釋放 NodeTable[i].adj = p->link; delete p; p = NodeTable[i].adj; } } delete [ ] NodeTable; //釋放頂點(diǎn)表},鄰接表部分成員
34、函數(shù)的實(shí)現(xiàn)template int Graph ::GetVertexPos ( const Type vertex ) {//根據(jù)頂點(diǎn)名vertex查找它在鄰接表中位置 for ( int i = 0; i int Graph ::GetFirstNeighbor ( int v ) {//查找頂點(diǎn) v 第一個(gè)鄰接頂點(diǎn)在鄰接表中位置,if ( v != -1 ) { //若頂點(diǎn)
35、存在 Edge * p = NodeTable[v].adj; if ( p != NULL ) return p->dest; } return -1; //若頂點(diǎn)不存在}template int Graph :: GetNextNeighbor ( int v, int w ) {//查找頂點(diǎn) v 在鄰接頂點(diǎn)
36、 w 后下一個(gè)鄰接頂點(diǎn) if ( v != -1 ) { Edge * p = NodeTable[v1].adj;,,while ( p != NULL ) { if ( p->dest == w && p->link != NULL ) return p->link->dest; //返回下一個(gè)鄰接頂點(diǎn)在鄰
37、接表中位置 else p = p->link; } } return -1; //沒有查到下一個(gè)鄰接頂點(diǎn)},template float Graph :: GetWeight ( int v1, int v2) { //取兩端點(diǎn)為 v1 和 v2 的邊上的權(quán)值 if ( v1 != -1 && v2 != -1 ) { Edge * p =
38、NodeTable[v1].adj; while ( p != NULL ) { if ( p->dest == v2 ) return p->cost; else p = p->link; } } return 0;},鄰接多重表 (Adjacency Multilist),在鄰接多重表中, 每一條邊只有一個(gè)邊結(jié)點(diǎn)。為有關(guān)邊的處理提供了方便。
39、無(wú)向圖的情形邊結(jié)點(diǎn)的結(jié)構(gòu),mark vertex1 vertex2 path1 path2,,,,,其中, mark 是記錄是否處理過的標(biāo)記; vertex1和vertex2是該邊兩頂點(diǎn)位置。path1域是鏈接指針, 指向下一條依附頂點(diǎn)vertex1的邊;path2 是指向下一條依附頂點(diǎn)vertex2的邊鏈接指針。,頂點(diǎn)結(jié)點(diǎn)的結(jié)構(gòu) 存儲(chǔ)頂點(diǎn)信息的結(jié)點(diǎn)表以順序表方式組織, 每一個(gè)頂點(diǎn)結(jié)點(diǎn)有兩個(gè)數(shù)據(jù)成員:其中,da
40、ta 存放與該頂點(diǎn)相關(guān)的信息,F(xiàn)irstout 是指示第一條依附該頂點(diǎn)的邊的指針。在鄰接多重表中, 所有依附同一個(gè)頂點(diǎn)的邊都鏈接在同一個(gè)單鏈表中。,data Firstout,,需要時(shí)還可設(shè)置一個(gè)存放與該邊相關(guān)的權(quán)值的域 cost。,從頂點(diǎn) i 出發(fā), 可以循鏈找到所有依附于該頂點(diǎn)的邊,也可以找到它的所有鄰接頂點(diǎn)。有向圖的情形在用鄰接表表示有向圖時(shí), 有時(shí)需要同時(shí)使用鄰接表和逆鄰接表。用有向圖的鄰接多重表(十字鏈表
41、)可把兩個(gè)表結(jié)合起來表示。邊結(jié)點(diǎn)的結(jié)構(gòu),mark vertex1 vertex2 path1 path2,,,,,其中,mark是處理標(biāo)記;vertex1和vertex2指明該有向邊始頂點(diǎn)和終頂點(diǎn)的位置。,頂點(diǎn)結(jié)點(diǎn)的結(jié)構(gòu) 每個(gè)頂點(diǎn)有一個(gè)結(jié)點(diǎn),它相當(dāng)于出邊表和入邊表的表頭結(jié)點(diǎn):其中,數(shù)據(jù)成員data存放與該頂點(diǎn)相關(guān)的信息,指針Firstin指示以該頂點(diǎn)為始頂點(diǎn)的出邊表的第一條邊,F(xiàn)irstout指示以該頂點(diǎn)為終頂
42、點(diǎn)的入邊表的第一條邊。,data Firstin Firstout,,,path1是指向始頂點(diǎn)與該邊相同的下一條邊的指針;path2是指向終頂點(diǎn)與該邊相同的下一條邊的指針。需要時(shí)還可有權(quán)值域cost。,,,,鄰接多重表的結(jié)構(gòu),,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,mark vtx1 vtx2 path1 path2,,,,,,,?,?,? ?,? ?,? ?,? ?,0 1,0
43、 3,1 2,2 3,3 4,4 0,e1,e2,e3,e4,e5,e6,data Fin Fout,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,A,B,C,D,E,0,1,2,3,4,,,,,,,,,,e1,e2,e3,e4,e5,e6,A,B,C,D,E,圖的遍歷與連通性,從已給的連通圖中某一頂點(diǎn)出發(fā),沿著一 些邊訪遍圖中所有的頂點(diǎn),且使每個(gè)頂點(diǎn)僅被訪問一次,就叫做圖
44、的遍歷 ( Graph Traversal )。圖中可能存在回路,且圖的任一頂點(diǎn)都可能與其它頂點(diǎn)相通,在訪問完某個(gè)頂點(diǎn)之后可能會(huì)沿著某些邊又回到了曾經(jīng)訪問過的頂點(diǎn)。為了避免重復(fù)訪問,可設(shè)置一個(gè)標(biāo)志頂點(diǎn)是否被訪問過的輔助數(shù)組 visited[ ]。,輔助數(shù)組 visited[ ] 的初始狀態(tài)為 0, 在圖的遍歷過程中, 一旦某一個(gè)頂點(diǎn) i 被訪問, 就立即讓 visited[i] 為 1, 防止它被多次訪問。圖的遍歷的分類:深度優(yōu)
45、先搜索 DFS (Depth First Search)廣度優(yōu)先搜索 BFS (Breadth First Search),,,,,,,,深度優(yōu)先搜索DFS ( Depth First Search ),深度優(yōu)先搜索的示例,,,,,,,A,C,D,E,,,,,G,B,F,I,,H,,,,,,,,,,A,C,D,E,,,,,G,B,F,I,,H,1,2,3,4,,,,,,,,,5,,,,,,,,,
46、6,7,8,9,1,2,3,4,5,6,7,8,9,前進(jìn),,回退,,深度優(yōu)先搜索過程 深度優(yōu)先生成樹,DFS 在訪問圖中某一起始頂點(diǎn) v 后, 由 v 出發(fā), 訪問它的任一鄰接頂點(diǎn) w1; 再?gòu)?w1 出發(fā),訪問與 w1鄰 接但還沒有訪問過的頂點(diǎn) w2; 然后再?gòu)?w2 出發(fā), 進(jìn)行類似的訪問, … 如此進(jìn)行下去, 直至到達(dá)所有的鄰接頂點(diǎn)都被訪問過的頂點(diǎn) u 為止。接著, 退回一步, 退到前一次剛訪問過
47、的頂點(diǎn), 看是否還有其它沒有被訪問的鄰接頂點(diǎn)。如果有, 則訪問此頂點(diǎn), 之后再?gòu)拇隧旤c(diǎn)出發(fā), 進(jìn)行與前述類似的訪問; 如果沒有, 就再退回一步進(jìn)行搜索。重復(fù)上述過程, 直到連通圖中所有頂點(diǎn)都被訪問過為止。,圖的深度優(yōu)先搜索算法template void Graph :: DFS ( ) { int * visited = new int [NumVertices]; for ( int i = 0; i
48、 void Graph ::DFS ( const int v, int visited [ ] ) {,cout << GetValue (v) << ‘ ’; //訪問頂點(diǎn) v visited[v] = 1; //頂點(diǎn) v 作訪問標(biāo)記 int w = GetFirstNeighbor (v); //取 v 的第一個(gè)鄰接頂點(diǎn) w while ( w !=
49、 -1 ) { //若鄰接頂點(diǎn) w 存在 if ( !visited[w] ) DFS ( w, visited ); //若頂點(diǎn) w 未訪問過, 遞歸訪問頂點(diǎn) w w = GetNextNeighbor ( v, w ); //取頂點(diǎn) v 排在 w 后的下一個(gè)鄰接頂點(diǎn) }},,廣度優(yōu)先搜索BFS ( Breadth First Search ),廣
50、度優(yōu)先搜索的示例,,,,,,,,,,,,,A,C,D,E,,,,,G,B,F,I,,H,,,,,,,,,,A,C,D,E,,,,G,B,F,,H,1,2,3,4,,,5,,,,,,,6,7,8,9,1,2,3,4,5,6,7,8,9,廣度優(yōu)先搜索過程 廣度優(yōu)先生成樹,,,,,,,,,I,BFS在訪問了起始頂點(diǎn) v 之后, 由 v 出發(fā), 依次訪問 v 的各個(gè)未被訪問過的鄰接頂點(diǎn) w1, w2, …, wt, 然
51、后再順序訪問 w1, w2, …, wt 的所有還未被訪問過的鄰接頂點(diǎn)。再?gòu)倪@些訪問過的頂點(diǎn)出發(fā),再訪問它們的所有還未被訪問過的鄰接頂點(diǎn),… 如此做下去,直到圖中所有頂點(diǎn)都被訪問到為止。廣度優(yōu)先搜索是一種分層的搜索過程, 每向前走一步可能訪問一批頂點(diǎn), 不像深度優(yōu)先搜索那樣有往回退的情況。因此, 廣度優(yōu)先搜索不是一個(gè)遞歸的過程。,為了實(shí)現(xiàn)逐層訪問, 算法中使用了一個(gè)隊(duì)列,以記憶正在訪問的這一層和上一層的頂點(diǎn),以便于向下一層訪問。為避
52、免重復(fù)訪問, 需要一個(gè)輔助數(shù)組 visited [ ],給被訪問過的頂點(diǎn)加標(biāo)記。,圖的廣度優(yōu)先搜索算法template void Graph ::BFS ( int v ) { int * visited = new int[NumVertices]; for ( int i = 0; i < NumVertices; i++ ) visited[i] = 0; //visit
53、ed初始化,cout q; q.EnQueue (v); //進(jìn)隊(duì)列 while ( !q.IsEmpty ( ) ) { //隊(duì)空搜索結(jié)束 q.GetFront ( v ); q.DeQueue ( ); int w = GetFirstNeighbor (v); while ( w != -1 ) { //若鄰接頂點(diǎn) w 存在 if
54、 ( !visited[w] ) { //未訪問過 cout << GetValue (w) << ‘ ’; visited[w] = 1; q.EnQueue (w); } w = GetNextNeighbor (v, w); } //重復(fù)檢測(cè) v 的所有鄰接頂點(diǎn),}
55、 //外層循環(huán),判隊(duì)列空否 delete [ ] visited; },連通分量 (Connected component),當(dāng)無(wú)向圖為非連通圖時(shí), 從圖中某一頂點(diǎn)出發(fā), 利用深度優(yōu)先搜索算法或廣度優(yōu)先搜索算法不可能遍歷到圖中的所有頂點(diǎn), 只能訪問到該頂點(diǎn)所在的最大連通子圖(連通分量)的所有頂點(diǎn)。,,若從無(wú)向圖的每一個(gè)連通分量中的一個(gè)頂點(diǎn)出發(fā)進(jìn)行遍歷, 可求得無(wú)向圖的所有連通分量。在算法中, 需要對(duì)圖的每一個(gè)
56、頂點(diǎn)進(jìn)行檢測(cè):若已被訪問過,則該頂點(diǎn)一定是落在圖中已求得的連通分量上;若還未被訪問,則從該頂點(diǎn)出發(fā)遍歷圖,可求得圖的另一個(gè)連通分量。對(duì)于非連通的無(wú)向圖,所有連通分量的生成樹組成了非連通圖的生成森林。,,,,,,,,,,,,,,,,,,,,,,,A,C,D,E,,,,I,B,F,O,,G,,H,,J,,N,,M,,L,,K,非連通無(wú)向圖,,,,,,,,,,,,,,A,,H,,K,,,,,C,D,E,,,,I,B,F,O,,G,,J,,N
57、,,M,,L,,,,,,,,,,,,,非連通圖的連通分量,確定連通分量的算法 template void Graph ::Components ( ) { int *visited = new int[NumVertices]; for ( int i = 0; i < NumVertices; i++ ) visited[i] = 0; //v
58、isited 初始化 for ( i = 0; i < NumVertices; i++ ) if ( !visited[i] ) { //檢測(cè)頂點(diǎn)是否訪問過 DFS ( i, visited ); //未訪問, 出發(fā)訪問 OutputNewComponent ( ); //連通分量 },,,,,,,,,,,【例】以深度優(yōu)先搜索方法從頂點(diǎn) 出發(fā)遍歷圖, 建立深度優(yōu)先生
59、成森林。,A,B,C,D,E,F,G,,,,,,A,A,B,D,E,C,F,G,有向圖,深度優(yōu)先生成森林,,,,,,,delete [ ] visited; },template void Graph ::DFS_Forest ( Tree &T ) { TreeNode * rt, * subT; int *visited = new int[n]; //
60、創(chuàng)建訪問數(shù)組 for ( int i = 0; i < n; i++ ) visited [i] = 0; //初始化 for ( i = 0; i < n; i++ ) //遍歷所有頂點(diǎn) if ( !visited[i] ) { //頂點(diǎn) i 未訪問過 if ( T.IsEmpty ( ) ) //原
61、為空森林,建根 subT = rt = T.BuildRoot ( GetValue(i) ); //頂點(diǎn) i 的值成為根 rt 的值 else,subT = T.InsertRightSibling ( subT, GetValue(i) ); //頂點(diǎn) i 的值成為 subT 右兄弟的值 DFS_Tree ( T,
62、subT, i, visited ); //從頂點(diǎn) i 出發(fā)深度優(yōu)先遍歷, 建立以 // subT為根的 T 的子樹 }}template void Graph :: DFS_Tree ( Tree &T, TreeNode *RT, int v, int visited [ ] ) {,TreeNode * p; visited [v] = 1; //
63、頂點(diǎn) v 作訪問過標(biāo)志 int w = GetFirstNeighbor (v); //取頂點(diǎn) v 的第一個(gè)鄰接頂點(diǎn) w int FirstChild = 1; //第一個(gè)未訪問子女應(yīng)是 v 的左子女 while ( w ) { //鄰接頂點(diǎn) w 存在 if ( ! visited [w] ) { // w未訪問過, 將成為 v 的子女
64、 if ( FirstChild ) { p = T.InsertLeftChild ( RT,GetValue(w) ); // p 插入為 RT 的左子女,FirstChild = 0; //建右兄弟 } else p = T.InsertRightSibling ( p, GetValue(w) );
65、 // p 插入為 p 的左子女 DFS_Tree ( T, p, w, visited ); //遞歸建立 w 的以 p 為根的子樹 } //鄰接頂點(diǎn) w 處理完 w = GetNextNeighbor ( v, w ); //取 v 的下一個(gè)鄰接頂點(diǎn) w } //回到 while 判鄰接頂點(diǎn)
66、w 存在},,重連通分量 (Biconnected Component),在無(wú)向連通圖G中, 當(dāng)且僅當(dāng)刪去G中的頂點(diǎn)v及所有依附于v的所有邊后, 可將圖分割成兩個(gè)或兩個(gè)以上的連通分量,則稱頂點(diǎn)v為關(guān)節(jié)點(diǎn)。沒有關(guān)節(jié)點(diǎn)的連通圖叫做重連通圖。在重連通圖上, 任何一對(duì)頂點(diǎn)之間至少存在有兩條路徑, 在刪去某個(gè)頂點(diǎn)及與該頂點(diǎn)相關(guān)聯(lián)的邊時(shí), 也不破壞圖的連通性。一個(gè)連通圖G如果不是重連通圖,那么它可以包括幾個(gè)重連通分量。,在一個(gè)無(wú)向連通圖G中
67、, 重連通分量可以利用深度優(yōu)先生成樹找到。在圖中各頂點(diǎn)旁標(biāo)明的深度優(yōu)先數(shù), 給出進(jìn)行深度優(yōu)先搜索時(shí)各頂點(diǎn)訪問的次序。深度優(yōu)先生成樹的根是關(guān)節(jié)點(diǎn)的充要條件是它至少有兩個(gè)子女。其它頂點(diǎn) u 是關(guān)節(jié)點(diǎn)的充要條件是它至少有一個(gè)子女 w, 從 w 出發(fā), 不能通過 w、w 的子孫及一條回邊所組成的路徑到達(dá) u 的祖先。,,,,,,,,,,,,,,,,,,,,,,,,A,B,C,D,E,F,G,H,I,J,,,,,,,,,,,,,,,,,,,
68、,,,A,B,C,D,E,F,G,H,J,,,,,,,,,,,,,,,,,,,,,A,B,C,D,E,F,G,H,J,I,I,1,2,3,4,5,6,7,8,9,10,,根有兩 個(gè)子女,,,關(guān)節(jié)點(diǎn),關(guān)節(jié)點(diǎn),,關(guān)節(jié)點(diǎn),,最小生成樹 ( minimum cost spanning tree ),使用不同的遍歷圖的方法,可以得到不同的生成樹;從不同的頂點(diǎn)出發(fā),也可能得到不同的生成樹。按照生成樹的定義,n 個(gè)頂點(diǎn)的連通網(wǎng)絡(luò)的生成樹有 n 個(gè)
69、頂點(diǎn)、n-1 條邊。構(gòu)造最小生成樹的準(zhǔn)則必須使用且僅使用該網(wǎng)絡(luò)中的n-1 條邊來聯(lián)結(jié)網(wǎng)絡(luò)中的 n 個(gè)頂點(diǎn);不能使用產(chǎn)生回路的邊;各邊上的權(quán)值的總和達(dá)到最小。,克魯斯卡爾 (Kruskal) 算法,克魯斯卡爾算法的基本思想: 設(shè)有一個(gè)有 n 個(gè)頂點(diǎn)的連通網(wǎng)絡(luò) N = { V, E }, 最初先構(gòu)造一個(gè)只有 n 個(gè)頂點(diǎn), 沒有邊的非連通圖 T = { V, ? }, 圖中每個(gè)頂點(diǎn)自成一個(gè)連通分量。當(dāng)在 E 中選到一
溫馨提示
- 1. 本站所有資源如無(wú)特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請(qǐng)下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請(qǐng)聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁(yè)內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 眾賞文庫(kù)僅提供信息存儲(chǔ)空間,僅對(duì)用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對(duì)用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對(duì)任何下載內(nèi)容負(fù)責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請(qǐng)與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對(duì)自己和他人造成任何形式的傷害或損失。
最新文檔
- 《數(shù)據(jù)結(jié)構(gòu)》講義
- 數(shù)據(jù)結(jié)構(gòu)講義
- 《數(shù)據(jù)結(jié)構(gòu)》講義
- 數(shù)據(jù)結(jié)構(gòu)講義第9章
- 學(xué)習(xí)指南(數(shù)據(jù)結(jié)構(gòu)基礎(chǔ))
- 學(xué)習(xí)數(shù)據(jù)結(jié)構(gòu)的意義和作用
- 學(xué)習(xí)數(shù)據(jù)結(jié)構(gòu)心得體會(huì)
- 學(xué)習(xí)數(shù)據(jù)結(jié)構(gòu)心得體會(huì)
- 算法與數(shù)據(jù)結(jié)構(gòu)各章學(xué)習(xí)要點(diǎn)
- 學(xué)習(xí)數(shù)據(jù)結(jié)構(gòu)心得體會(huì)
- 數(shù)據(jù)結(jié)構(gòu)
- 上海交通大學(xué)《數(shù)據(jù)結(jié)構(gòu)》重點(diǎn)講義各章節(jié)
- 上海交通大學(xué)《數(shù)據(jù)結(jié)構(gòu)》重點(diǎn)講義各章節(jié)
- 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計(jì)--數(shù)據(jù)結(jié)構(gòu)的實(shí)現(xiàn)
- 數(shù)據(jù)結(jié)構(gòu)論文數(shù)據(jù)結(jié)構(gòu)實(shí)驗(yàn)教學(xué)探索
- 《數(shù)據(jù)結(jié)構(gòu)》大綱
- 數(shù)據(jù)結(jié)構(gòu)答案
- 數(shù)據(jù)結(jié)構(gòu)(六)
- 《數(shù)據(jù)結(jié)構(gòu)》教案
- 數(shù)據(jù)結(jié)構(gòu)范本
評(píng)論
0/150
提交評(píng)論