圖的遍歷,所謂遍歷,便是對結點的訪問。一個圖有那麼多個結點,如何遍歷這些結點,須要特定策略,通常有兩種訪問策略:java
深度優先遍歷,從初始訪問結點出發,咱們知道初始訪問結點可能有多個鄰接結點,深度優先遍歷的策略就是首先訪問第一個鄰接結點,而後再以這個被訪問的鄰接結點做爲初始結點,訪問它的第一個鄰接結點。總結起來能夠這樣說:每次都在訪問完當前結點後首先訪問當前結點的第一個鄰接結點。 咱們從這裏能夠看到,這樣的訪問策略是優先往縱向挖掘深刻,而不是對一個結點的全部鄰接結點進行橫向訪問。 具體算法表述以下:算法
相似於一個分層搜索的過程,廣度優先遍歷須要使用一個隊列以保持訪問過的結點的順序,以便按這個順序來訪問這些結點的鄰接結點。 具體算法表述以下:segmentfault
package com.lifeibigdata.algorithms.graph.liantongfenliang; import java.util.LinkedList; import java.util.Queue; /** * Created by leofei.li on 2016/5/21. */ public class BFS { static int verNum; static boolean []visited; static String []ver={"A","B","C","D","E"}; static int [][]edge; void addEdge(int i,int j){ if(i == j)return; edge[i][j]=1; edge[j][i]=1; } void dfsTraverse(){ visited = new boolean[verNum]; for(int i = 0; i< verNum; i ++){ if(visited[i] == false){ dfs(i); } } } void dfs(int i){ visited[i] = true; System.out.print(ver[i] + " "); for(int j = 0; j < verNum; j++){ if(visited[j] == false && edge[i][j] == 1){ dfs(j); } } } void bfsTraverse(){ visited = new boolean[verNum]; Queue<Integer> quene = new LinkedList<Integer>(); for (int i = 0; i < verNum; i ++){ if(visited[i] == false){ visited[i] = true; System.out.print(ver[i]+" "); quene.add(i); //此處存儲的是索引 while (!quene.isEmpty()){ //注意結束條件 int j = quene.poll(); for (int k = 0; k < verNum; k++){ //找到該節點全部的子節點 if(edge[j][k] == 1 && visited[k] == false){ visited[k] = true; System.out.print(ver[k]+" "); quene.add(k); } } } } } } void con(){ int count = 0; visited = new boolean[verNum]; for(int i = 0; i < verNum; i ++){ if(!visited[i]){ count++; dfsTraverse(); } } System.out.println("共有"+count+"個連通份量!"); } public static void main(String[] args) { verNum = ver.length; edge = new int[verNum][verNum]; for(int i=0;i<verNum;i++){ for (int j=0;j<verNum;j++){ edge[i][j]=0; } } BFS b = new BFS(); b.addEdge(0, 3); b.addEdge(0, 4); b.addEdge(1, 2); b.addEdge(2, 4); b.addEdge(2, 3); System.out.println("圖的深度遍歷操做:"); b.dfsTraverse(); System.out.println(); System.out.println("圖的廣度遍歷操做:"); b.bfsTraverse(); System.out.println(); System.out.println("連通份量:"); b.con(); } }
無論深搜仍是廣搜都須要申請總節點個數長度的數組用來標記節點是否訪問過(訪問數組);深搜是遍歷訪問數組,從某個未被訪問的節點輻射出去,每次輻射結束後,判斷訪問數組中是否還有未被訪問的節點,若是有,再次輻射,若是沒有,深搜結束;廣搜是一樣是遍歷數組,判斷是否有爲被訪問的節點,若是有放到對列中,而後彈出隊列,將彈出節點全部未被訪問的鄰接節點放到隊列中,繼續進行訪問隊列,直到隊列爲空時繼續判斷訪問數組數組
原文連接:https://segmentfault.com/a/1190000002685939#articleHeader0code