<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
在生成樹的過程中,把已經在生成樹中的節點看作一個集合,把剩下的節點看作另外一個集合,從連線兩個集合的邊中選擇一條權值最小的邊即可。
首先任選一個節點,例如節點1,把它放在集合 U 中,U={1},那麼剩下的節點為 V-U={2,3,4,5,6,7},集合 V 是圖的所有節點集合。
現在只需要看看連線兩個集合(U 和 V-U)的邊中,哪一條邊的權值最小,把權值最小的邊關聯的節點加入集合 U 中。從上圖可以看出,連線兩個集合的 3 條邊中,1-2 邊的權值最小,選中它,把節點 2 加入集合 U 中,U={1,2},V - U={3,4,5,6},如下圖所示。
再從連線兩個集合(U 和 V-U)的邊中選擇一條權最小的邊。從上圖看出,在連線兩個集合的4條邊中,節點2到節點7的邊權值最小,選中這條邊,把節點7加入集合U={1,2,7}中,V-U={3,4,5,6}。
如此下去,直到 U=V 結束,選中的邊和所有的節點組成的圖就是最小生成樹。這就是 Prim 演演算法。
直觀地看圖,很容易找出集合 U 到 集合 U-V 的邊中哪條邊的權值是最小的,但在程式中窮舉這些邊,再找最小值,則時間複雜度太高。可以通過設定陣列巧妙解決這個問題,closet[j] 表示集合 V-U 中的節點 j 到集合 U 中的最鄰近點,lowcost[j] 表示集合 V-U 中節點 j 到集合 U 中最鄰近點的邊值,即邊(j,closest[j]) 的權值。
例如在上圖中,節點 7 到集合 U 中的最鄰近點是2,cloeest[7]=2。節點 7 到最鄰近點2 的邊值為1,即邊(2,7)的權值,記為 lowcost[7]=1,如下圖所示。
所以只需在集合 V - U 中找到 lowcost[] 只最小的節點即可。
1.初始化
令集合 U={u0},u0 屬於 V,並初始化陣列 closest[]、lowcost[]和s[]。
2.在集合 V-U 中找 lowcost 值最小的節點t,即 lowcost[t]=min{lowcost[j]},j 屬於 V-U,滿足該公式的節點 t 就是集合 V-U 中連線 U 的最鄰近點。
3.將節點 t 加入集合 U 中。
4.如果集合 V - U 為空,則演演算法結束,否則轉向步驟 5。
5.對集合 V-U 中的所有節點 j 都更新其 lowcost[] 和 closest[]。if(C[t][j]<lowcost[j]){lowcost[j]=C[t][j];closest[j]=t;},轉向步驟2。
按照上面步驟,最終可以得到一棵權值之和最小的生成樹。
圖 G=(V,E)是一個無向連通帶權圖,如下圖所示。
1 初始化。假設 u0=1,令集合 U={1},集合 V-U={2,3,4,5,6,7},s[1]=true,初始化陣列 closest[]:除了節點1,其餘節點均為1,表示集合 V-U 中的節點到集合 U 的最鄰近點均為1.lowcost[]:節點1到集合 V-U 中節點的邊值。closest[] 和 lowcost[] 如下圖所示。
初始化後的圖為:
2 找 lowcost 最小的節點,對應的 t=2,選中的邊和節點如下圖。
3 加入集合U中。將節點 t 加入集合 U 中,U={1,2},同時更新 V-U={3,4,5,6,7}
4 更新。對 t 在集合 V-U 中的每一個鄰接點 j,都可以藉助 t 更新。節點 2 的鄰接點是節點 3 和節點7。
C[2][3]=20<lowcost[3]=無窮大,更新最鄰近距離 lowcost[3]=20,最鄰近點 closest[3]=2;
C[2][7]=1<lowcost[7]=36,更新最鄰近距離 lowcost[7]=1,最鄰近點 closest[7]=2;
更新後的 closest[] 和 lowcost[] 如下圖所示。
更新後的集合如下圖所示:
5 找 lowcost 最小的節點,對應的 t=7,選中的邊和節點如下圖。
6 加入集合U中。將節點 t 加入集合 U 中,U={1,2,7},同時更新 V-U={3,4,5,6}
7 更新。對 t 在集合 V-U 中的每一個鄰接點 j,都可以藉助 t 更新。節點 7 的鄰接點是節點 3、4、5、6。
更新後的 closest[] 和 lowcost[] 如下圖所示。
更新後的集合如下圖所示:
8 找 lowcost 最小的節點,對應的 t=3,選中的邊和節點如下圖。
9 加入集合U中。將節點 t 加入集合 U 中,U={1,2,3,7},同時更新 V-U={4,5,6}
10 更新。對 t 在集合 V-U 中的每一個鄰接點 j,都可以藉助 t 更新。節點 3 的鄰接點是節點 4。
C[3][4]=15>lowcost[4]=9,不更新
closest[] 和 lowcost[] 陣列不改變。
更新後的集合如下圖所示:
11 找 lowcost 最小的節點,對應的 t=4,選中的邊和節點如下圖。
12 加入集合U中。將節點 t 加入集合 U 中,U={1,2,3,4,7},同時更新 V-U={5,6}
13 更新。對 t 在集合 V-U 中的每一個鄰接點 j,都可以藉助 t 更新。節點 4 的鄰接點是節點 5。
C[4][5]=3<lowcost[5]=16,更新最鄰近距離 lowcost[5]=3,最鄰近點 closest[5]=4;
更新後的 closest[] 和 lowcost[] 如下圖所示。
更新後的集合如下圖所示:
14 找 lowcost 最小的節點,對應的 t=5,選中的邊和節點如下圖。
15 加入集合U中。將節點 t 加入集合 U 中,U={1,2,3,4,5,7},同時更新 V-U={6}
16 更新。對 t 在集合 V-U 中的每一個鄰接點 j,都可以藉助 t 更新。節點 5 的鄰接點是節點 6。
C[5][6]=17<lowcost[6]=25,更新最鄰近距離 lowcost[6]=17,最鄰近點 closest[6]=5;
更新後的集合如下圖所示:
17 找 lowcost 最小的節點,對應的 t=6,選中的邊和節點如下圖。
18 加入集合U中。將節點 t 加入集合 U 中,U={1,2,3,4,5,6,7},同時更新 V-U={}
19 更新。對 t 在集合 V-U 中的每一個鄰接點 j,都可以藉助 t 更新。節點 6 在集合 V-U 中無鄰接點。不用更新 closest[] 和 lowcost[] 。
20 得到的最小生成樹如下。最小生成樹的權值之和為 57.
package graph.prim; import java.util.Scanner; public class Prim { static final int INF = 0x3f3f3f3f; static final int N = 100; // 如果s[i]=true,說明頂點i已加入U static boolean s[] = new boolean[N]; static int c[][] = new int[N][N]; static int closest[] = new int[N]; static int lowcost[] = new int[N]; static void Prim(int n) { // 初始時,集合中 U 只有一個元素,即頂點 1 s[1] = true; for (int i = 1; i <= n; i++) { if (i != 1) { lowcost[i] = c[1][i]; closest[i] = 1; s[i] = false; } else lowcost[i] = 0; } for (int i = 1; i < n; i++) { int temp = INF; int t = 1; // 在集合中 V-u 中尋找距離集合U最近的頂點t for (int j = 1; j <= n; j++) { if (!s[j] && lowcost[j] < temp) { t = j; temp = lowcost[j]; } } if (t == 1) break; // 找不到 t,跳出迴圈 s[t] = true; // 否則,t 加入集合U for (int j = 1; j <= n; j++) { // 更新 lowcost 和 closest if (!s[j] && c[t][j] < lowcost[j]) { lowcost[j] = c[t][j]; closest[j] = t; } } } } public static void main(String[] args) { int n, m, u, v, w; Scanner scanner = new Scanner(System.in); n = scanner.nextInt(); m = scanner.nextInt(); int sumcost = 0; for (int i = 1; i <= n; i++) for (int j = 1; j <= n; j++) c[i][j] = INF; for (int i = 1; i <= m; i++) { u = scanner.nextInt(); v = scanner.nextInt(); w = scanner.nextInt(); c[u][v] = c[v][u] = w; } Prim(n); System.out.println("陣列lowcost:"); for (int i = 1; i <= n; i++) System.out.print(lowcost[i] + " "); System.out.println(); for (int i = 1; i <= n; i++) sumcost += lowcost[i]; System.out.println("最小的花費:" + sumcost); } }
以上就是Java中Prime演演算法的原理與實現詳解的詳細內容,更多關於Java Prime演演算法的資料請關注it145.com其它相關文章!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45