最小生成樹之prim算法


MST(Minimum Spanning Tree,最小生成樹)問題有兩種通用的解法,Prim算法就是其中之一,它是從點的方面考慮構建一顆MST,大致思想是:設圖G頂點集合為U,首先任意選擇圖G中的一點作為起始點a,將該點加入集合V,再從集合U-V中找到另一點b使得點b到V中任意一點的權值最小,此時將b點也加入集合V;以此類推,現在的集合V={a,b},再從集合U-V中找到另一點c使得點c到V中任意一點的權值最小,此時將c點加入集合V,直至所有頂點全部被加入V,此時就構建出了一顆MST。因為有N個頂點,所以該MST就有N-1條邊,每一次向集合V中加入一個點,就意味着找到一條MST的邊。

Prim算法適用於稠密圖 Kruskal適用於稀疏圖

用圖示和代碼說明:

初始狀態:


設置2個數據結構

lowcost[i]:表示以i為終點的邊的最小權值,當lowcost[i]=0說明以i為終點的邊的最小權值=0,也就是表示i點加入了MST

mst[i]:表示對應lowcost[i]的起點,即說明邊<mst[i],i>是MST的一條邊,當mst[i]=0表示起點i加入MST


我們假設V1是起始點,進行初始化(*代表無限大,即無通路):


lowcost[2]=6lowcost[3]=1lowcost[4]=5lowcost[5]=*,lowcost[6]=*

mst[2]=1mst[3]=1,mst[4]=1mst[5]=1,mst[6]=1(所有點默認起點是V1)


明顯看出,以V3為終點的邊的權值最小=1,所以邊<mst[3],3>=1加入MST


此時,因為點V3的加入,需要更新lowcost數組和mst數組:

 

lowcost[2]=5lowcost[3]=0lowcost[4]=5lowcost[5]=6,lowcost[6]=4

mst[2]=3mst[3]=0,mst[4]=1mst[5]=3,mst[6]=3


明顯看出,以V6為終點的邊的權值最小=4,所以邊<mst[6],6>=4加入MST


 

此時,因為點V6的加入,需要更新lowcost數組和mst數組:

 

lowcost[2]=5lowcost[3]=0lowcost[4]=2lowcost[5]=6lowcost[6]=0

mst[2]=3mst[3]=0,mst[4]=6mst[5]=3,mst[6]=0

 


明顯看出,以V4為終點的邊的權值最小=2,所以邊<mst[4],4>=4加入MST


 

此時,因為點V4的加入,需要更新lowcost數組和mst數組:

 

lowcost[2]=5,lowcost[3]=0,lowcost[4]=0,lowcost[5]=6lowcost[6]=0

mst[2]=3,mst[3]=0,mst[4]=0mst[5]=3mst[6]=0


明顯看出,以V2為終點的邊的權值最小=5,所以邊<mst[2],2>=5加入MST


 

此時,因為點V2的加入,需要更新lowcost數組和mst數組:

 

lowcost[2]=0,lowcost[3]=0,lowcost[4]=0,lowcost[5]=3,lowcost[6]=0

mst[2]=0,mst[3]=0,mst[4]=0mst[5]=2mst[6]=0


很明顯,以V5為終點的邊的權值最小=3,所以邊<mst[5],5>=3加入MST

 

lowcost[2]=0,lowcost[3]=0lowcost[4]=0,lowcost[5]=0lowcost[6]=0

mst[2]=0,mst[3]=0mst[4]=0,mst[5]=0mst[6]=0


至此,MST構建成功,如圖所示:


 

 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstring>
 4 #include<algorithm>
 5 #include<cmath>
 6 #include<queue>
 7 #include<stack>
 8 #include<map>
 9 #include<sstream>
10 using namespace std;
11 typedef long long ll;
12 const int maxn = 2e3 + 10;
13 const int INF = 1 << 30;
14 int dir[4][2] = {1,0,0,1,-1,0,0,-1};
15 int T, n, m, x;
16 int Map[maxn][maxn];//存圖
17 int lowcost[maxn], mst[maxn];
18 void prim(int u)//最小生成樹起點
19 {
20     int sum_mst = 0;//最小生成樹權值
21     for(int i = 1; i <= n; i++)//初始化兩個數組
22     {
23         lowcost[i] = Map[u][i];
24         mst[i] = u;
25     }
26     mst[u] = -1;//設置成-1表示已經加入mst
27     for(int i = 1; i < n; i++)//此處只需要迭代n-1次即可
28     {
29         int minn = INF;
30         int v = -1;
31         //在lowcost數組中尋找未加入mst的最小值
32         for(int j = 1; j <= n; j++)
33         {
34             if(mst[j] != -1 && lowcost[j] < minn)
35             {
36                 v = j;
37                 minn = lowcost[j];
38             }
39         }
40         if(v != -1)//v=-1表示未找到最小的邊,
41         {//v表示當前距離mst最短的點
42             printf("%d %d %d\n", mst[v], v, lowcost[v]);//輸出路徑
43             mst[v] = -1;
44             sum_mst += lowcost[v];
45             for(int j = 1; j <= n; j++)//更新最短邊
46             {
47                 if(mst[j] != -1 && lowcost[j] > Map[v][j])
48                 {
49                     lowcost[j] = Map[v][j];
50                     mst[j] = v;
51                 }
52             }
53         }
54     }
55     printf("weight of mst is %d\n", sum_mst);
56 }
57 int main()
58 {
59     cin >> n >> m;
60     memset(Map, 0, sizeof(Map));
61     for(int i = 1; i <= m; i++)
62     {
63         int u, v, w;
64         cin >> u >> v >> w;
65         Map[u][v] = Map[v][u] = w;
66     }
67     for(int i = 1; i <= n; i++)
68     {
69         for(int j = 1; j <= n; j++)
70         {
71             if(i == j)Map[i][j] = 0;
72             else if(!Map[i][j])Map[i][j] = INF;
73         }
74     }
75     prim(1);
76     return 0;
77 }

輸入:

7 9
1 2 28
1 6 10
2 3 16
2 7 14
3 4 12
4 5 22
4 7 18
5 6 25
5 7 24

輸出:

1 6 10
6 5 25
5 4 22
4 3 12
3 2 16
2 7 14
weight of mst is 99

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM