轉自:http://blog.csdn.net/wang2534499/article/details/47750321
算法8:巧妙的鄰接表(數組實現)
之前我們介紹過圖的鄰接矩陣存儲法,它的空間和時間複雜度都是N2,現在我來介紹另外一種存儲圖的方法:鄰接表,這樣空間和時間複雜度就都是M。對於稀疏圖來說,M要遠遠小於N2。先上數據,如下。
- 4 5
- 1 4 9
- 4 3 8
- 1 2 5
- 2 4 6
- 1 3 7
第一行兩個整數n m。n表示頂點個數(頂點編號爲1~n),m表示邊的條數。接下來m行表示,每行有3個數x y z,表示頂點x到頂點y的邊的權值爲z。下圖就是一種使用鏈表來實現鄰接表的方法。
上面這種實現方法爲圖中的每一個頂點(左邊部分)都建立了一個單鏈表(右邊部分)。這樣我們就可以通過遍歷每個頂點的鏈表,從而得到該頂點所有的邊了。使用鏈表來實現鄰接表對於痛恨指針的的朋友來說,這簡直就是噩夢。這裏我將爲大家介紹另一種使用數組來實現的鄰接表,這是一種在實際應用中非常容易實現的方法。這種方法爲每個頂點i(i從1~n)也都保存了一個類似“鏈表”的東西,裏面保存的是從頂點i出發的所有的邊,具體如下。
首先我們按照讀入的順序爲每一條邊進行編號(1~m)。比如第一條邊“1 4 9”的編號就是1,“1 3 7”這條邊的編號是5。
這裏用u、v和w三個數組用來記錄每條邊的具體信息,即u[i]、v[i]和w[i]表示第i條邊是從第u[i]號頂點到v[i]號頂點(u[i]àv[i]),且權值爲w[i]。
再用一個first數組來存儲每個頂點其中一條邊的編號。以便待會我們來枚舉每個頂點所有的邊(你可能會問:存儲其中一條邊的編號就可以了?不可能吧,每個頂點都需要存儲其所有邊的編號才行吧!甭着急,繼續往下看)。比如1號頂點有一條邊是 “1 4 9”(該條邊的編號是1),那麼就將first[1]的值設爲1。如果某個頂點i沒有以該頂點爲起始點的邊,則將first[i]的值設爲-1。現在我們來看看具體如何操作,初始狀態如下。
咦?上圖中怎麼多了一個next數組,有什麼作用呢?不着急,待會再解釋,現在先讀入第一條邊“1 4 9”。
讀入第1條邊(1 4 9),將這條邊的信息存儲到u[1]、v[1]和w[1]中。同時爲這條邊賦予一個編號,因爲這條邊是最先讀入的,存儲在u、v和w數組下標爲1的單元格中,因此編號就是1。這條邊的起始點是1號頂點,因此將first[1]的值設爲1。
另外這條“編號爲1的邊”是以1號頂點(即u[1])爲起始點的第一條邊,所以要將next[1]的值設爲-1。也就是說,如果當前這條“編號爲i的邊”,是我們發現的以u[i]爲起始點的第一條邊,就將next[i]的值設爲-1(貌似的這個next數組很神祕啊⊙_⊙)。
讀入第2條邊(4 3 8),將這條邊的信息存儲到u[2]、v[2]和w[2]中,這條邊的編號爲2。這條邊的起始頂點是4號頂點,因此將first[4]的值設爲2。另外這條“編號爲2的邊”是我們發現以4號頂點爲起始點的第一條邊,所以將next[2]的值設爲-1。
讀入第3條邊(1 2 5),將這條邊的信息存儲到u[3]、v[3]和w[3]中,這條邊的編號爲3,起始頂點是1號頂點。我們發現1號頂點已經有一條“編號爲1 的邊”了,如果此時將first[1]的值設爲3,那“編號爲1的邊”豈不是就丟失了?我有辦法,此時只需將next[3]的值設爲1即可。現在你知道next數組是用來做什麼的吧。next[i]存儲的是“編號爲i的邊”的“前一條邊”的編號。
讀入第4條邊(2 4 6),將這條邊的信息存儲到u[4]、v[4]和w[4]中,這條邊的編號爲4,起始頂點是2號頂點,因此將first[2]的值設爲4。另外這條“編號爲4的邊”是我們發現以2號頂點爲起始點的第一條邊,所以將next[4]的值設爲-1。
讀入第5條邊(1 3 7),將這條邊的信息存儲到u[5]、v[5]和w[5]中,這條邊的編號爲5,起始頂點又是1號頂點。此時需要將first[1]的值設爲5,並將next[5]的值改爲3。
此時,如果我們想遍歷1號頂點的每一條邊就很簡單了。1號頂點的其中一條邊的編號存儲在first[1]中。其餘的邊則可以通過next數組尋找到。請看下圖。
細心的同學會發現,此時遍歷邊某個頂點邊的時候的遍歷順序正好與讀入時候的順序相反。因爲在爲每個頂點插入邊的時候都直接插入“鏈表”的首部而不是尾部。不過這並不會產生任何問題,這正是這種方法的其妙之處。
創建鄰接表的代碼如下。
- int n,m,i;
- //u、v和w的數組大小要根據實際情況來設置,要比m的最大值要大1
- int u[6],v[6],w[6];
- //first和next的數組大小要根據實際情況來設置,要比n的最大值要大1
- int first[5],next[5];
- scanf("%d %d",&n,&m);
- //初始化first數組下標1~n的值爲-1,表示1~n頂點暫時都沒有邊
- for(i=1;i<=n;i++)
- first[i]=-1;
- for(i=1;i<=m;i++)
- {
- scanf("%d %d %d",&u[i],&v[i],&w[i]);//讀入每一條邊
- //下面兩句是關鍵啦
- next[i]=first[u[i]];
- first[u[i]]=i;
- }
接下來如何遍歷每一條邊呢?我們之前說過其實first數組存儲的就是每個頂點i(i從1~n)的第一條邊。比如1號頂點的第一條邊是編號爲5的邊(1 3 7),2號頂點的第一條邊是編號爲4的邊(2 4 6),3號頂點沒有出向邊,4號頂點的第一條邊是編號爲2的邊(2 4 6)。那麼如何遍歷1號頂點的每一條邊呢?也很簡單。請看下圖:
遍歷1號頂點所有邊的代碼如下。
- k=first[1];// 1號頂點其中的一條邊的編號(其實也是最後讀入的邊)
- while(k!=-1) //其餘的邊都可以在next數組中依次找到
- {
- printf("%d %d %d\n",u[k],v[k],w[k]);
- k=next[k];
- }
遍歷每個頂點的所有邊的代碼如下。
- for(i=1;i<=n;i++)
- {
- k=first[i];
- while(k!=-1)
- {
- printf("%d %d %d\n",u[k],v[k],w[k]);
- k=next[k];
- }
- }
可以發現使用鄰接表來存儲圖的時間空間複雜度是O(M),遍歷每一條邊的時間複雜度是也是O(M)。如果一個圖是稀疏圖的話,M要遠小於N2。因此稀疏圖選用鄰接表來存儲要比鄰接矩陣來存儲要好很多。