亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C > 正文

C語言實現最小生成樹構造算法

2020-01-26 13:33:42
字體:
來源:轉載
供稿:網友

最小生成樹

最小生成樹(minimum spanning tree)是由n個頂點,n-1條邊,將一個連通圖連接起來,且使權值最小的結構。
最小生成樹可以用Prim(普里姆)算法或kruskal(克魯斯卡爾)算法求出。

我們將以下面的帶權連通圖為例講解這兩種算法的實現:

注:由于測試輸入數據較多,程序可以采用文件輸入

Prim(普里姆)算法

時間復雜度:O(N^2)(N為頂點數)
prim算法又稱“加點法”,用于邊數較多的帶權無向連通圖
方法:每次找與之連線權值最小的頂點,將該點加入最小生成樹集合中
注意:相同權值任選其中一個即可,但是不允許出現閉合回路的情況。 

 

代碼部分通過以下步驟可以得到最小生成樹:

1.初始化:

lowcost[i]:表示以i為終點的邊的最小權值,當lowcost[i]=0表示i點加入了MST。
mst[i]:表示對應lowcost[i]的起點,當mst[i]=0表示起點i加入MST。
由于我們規定最開始的頂點是1,所以lowcost[1]=0,MST[1]=0。即只需要對2~n進行初始化即可。

#define MAX 100 #define MAXCOST 0x7fffffff int graph[MAX][MAX]; void prim(int graph[][MAX], int n) {  int lowcost[MAX];  int mst[MAX];  int i, j, min, minid, sum = 0;  for (i = 2; i <= n; i++)  {  lowcost[i] = graph[1][i];//lowcost存放頂點1可達點的路徑長度  mst[i] = 1;//初始化以1位起始點  }  mst[1] = 0; 

2.查找最小權值及路徑更新

定義一個最小權值min和一個最小頂點ID minid,通過循環查找出min和minid,另外由于規定了某一頂點如果被連入,則lowcost[i]=0,所以不需要擔心重復點問題。所以找出的終點minid在MST[i]中可以找到對應起點,min為權值,直接輸出即可。
我們連入了一個新的頂點,自然需要對這一點可達的路徑及權值進行更新,所以循環中還應該包括路徑更新的代碼。

for (i = 2; i <= n; i++)  {  min = MAXCOST;  minid = 0;  for (j = 2; j <= n; j++)  {  if (lowcost[j] < min && lowcost[j] != 0)  {  min = lowcost[j];//找出權值最短的路徑長度  minid = j; //找出最小的ID  }  }  printf("V%d-V%d=%d/n",mst[minid],minid,min);  sum += min;//求和  lowcost[minid] = 0;//該處最短路徑置為0  for (j = 2; j <= n; j++) {  if (graph[minid][j] < lowcost[j])//對這一點直達的頂點進行路徑更新  {  lowcost[j] = graph[minid][j];  mst[j] = minid; }  }  }  printf("最小權值之和=%d/n",sum);} 

具體代碼如下:

#include<stdio.h> #define MAX 100 #define MAXCOST 0x7fffffff int graph[MAX][MAX]; void prim(int graph[][MAX], int n) {  int lowcost[MAX];  int mst[MAX];  int i, j, min, minid, sum = 0;  for (i = 2; i <= n; i++)  {  lowcost[i] = graph[1][i];//lowcost存放頂點1可達點的路徑長度  mst[i] = 1;//初始化以1位起始點  }  mst[1] = 0;  for (i = 2; i <= n; i++)  {  min = MAXCOST;  minid = 0;  for (j = 2; j <= n; j++)  {  if (lowcost[j] < min && lowcost[j] != 0)  {  min = lowcost[j];//找出權值最短的路徑長度  minid = j; //找出最小的ID  }  }  printf("V%d-V%d=%d/n",mst[minid],minid,min);  sum += min;//求和  lowcost[minid] = 0;//該處最短路徑置為0  for (j = 2; j <= n; j++) {  if (graph[minid][j] < lowcost[j])//對這一點直達的頂點進行路徑更新  {  lowcost[j] = graph[minid][j];  mst[j] = minid; }  }  }  printf("最小權值之和=%d/n",sum);} int main() {  int i, j, k, m, n;  int x, y, cost;  //freopen("1.txt","r",stdin);//文件輸入  scanf("%d%d",&m,&n);//m=頂點的個數,n=邊的個數  for (i = 1; i <= m; i++)//初始化圖  {  for (j = 1; j <= m; j++)  {  graph[i][j] = MAXCOST;  }  }  for (k = 1; k <= n; k++)  {  scanf("%d%d%d",&i,&j,&cost); graph[i][j] = cost;  graph[j][i] = cost;  }  prim(graph, m);  return 0; } 

編譯運行結果:

普里姆結果

kruskal(克魯斯卡爾)算法

時間復雜度:O(NlogN)(N為邊數)
kruskal算法又稱“加邊法”,用于邊數較少的稀疏圖
方法:每次找圖中權值最小的邊,將邊連接的兩個頂點加入最小生成樹集合中
注意:相同權值任選其中一個即可,但是不允許出現閉合回路的情況。

代碼部分通過以下步驟可以得到最小生成樹:

1.初始化:

構建邊的結構體,包括起始頂點、終止頂點,邊的權值
借用一個輔助數組vset[i]用來判斷某邊是否加入了最小生成樹集合

#define MAXE 100#define MAXV 100typedef struct{ int vex1; //邊的起始頂點 int vex2; //邊的終止頂點 int weight; //邊的權值}Edge;void kruskal(Edge E[],int n,int e){  int i,j,m1,m2,sn1,sn2,k,sum=0; int vset[n+1]; for(i=1;i<=n;i++) //初始化輔助數組 vset[i]=i; k=1;//表示當前構造最小生成樹的第k條邊,初值為1 j=0;//E中邊的下標,初值為0

2.取邊和輔助集合更新

按照排好的順序依次取邊,若不屬于同一集合則將其加入最小生成樹集合,每當加入新的邊,所連接的兩個點即納入最小生成樹集合,為避免重復添加,需要進行輔助集合更新
注:由于kruskal算法需要按照權值大小順序取邊,所以應該事先對圖按權值升序,這里我采用了快速排序算法,具體算法可以參照快速排序(C語言)

 while(k<e)//生成的邊數小于e時繼續循環 { m1=E[j].vex1; m2=E[j].vex2;//取一條邊的兩個鄰接點 sn1=vset[m1]; sn2=vset[m2];  //分別得到兩個頂點所屬的集合編號 if(sn1!=sn2)//兩頂點分屬于不同的集合,該邊是最小生成樹的一條邊 {//防止出現閉合回路  printf("V%d-V%d=%d/n",m1,m2,E[j].weight); sum+=E[j].weight; k++; //生成邊數增加 if(k>=n) break; for(i=1;i<=n;i++) //兩個集合統一編號 if (vset[i]==sn2) //集合編號為sn2的改為sn1 vset[i]=sn1; } j++; //掃描下一條邊 } printf("最小權值之和=%d/n",sum);}

具體算法實現:

#include <stdio.h>#define MAXE 100#define MAXV 100typedef struct{ int vex1; //邊的起始頂點 int vex2; //邊的終止頂點 int weight; //邊的權值}Edge;void kruskal(Edge E[],int n,int e){  int i,j,m1,m2,sn1,sn2,k,sum=0; int vset[n+1]; for(i=1;i<=n;i++) //初始化輔助數組 vset[i]=i; k=1;//表示當前構造最小生成樹的第k條邊,初值為1 j=0;//E中邊的下標,初值為0 while(k<e)//生成的邊數小于e時繼續循環 { m1=E[j].vex1; m2=E[j].vex2;//取一條邊的兩個鄰接點 sn1=vset[m1]; sn2=vset[m2];  //分別得到兩個頂點所屬的集合編號 if(sn1!=sn2)//兩頂點分屬于不同的集合,該邊是最小生成樹的一條邊 {//防止出現閉合回路  printf("V%d-V%d=%d/n",m1,m2,E[j].weight); sum+=E[j].weight; k++; //生成邊數增加  if(k>=n) break; for(i=1;i<=n;i++) //兩個集合統一編號 if (vset[i]==sn2) //集合編號為sn2的改為sn1 vset[i]=sn1; } j++; //掃描下一條邊 } printf("最小權值之和=%d/n",sum);}int fun(Edge arr[],int low,int high) { int key; Edge lowx; lowx=arr[low]; key=arr[low].weight; while(low<high) { while(low<high && arr[high].weight>=key) high--; if(low<high) arr[low++]=arr[high]; while(low<high && arr[low].weight<=key) low++; if(low<high) arr[high--]=arr[low]; } arr[low]=lowx; return low; } void quick_sort(Edge arr[],int start,int end){ int pos; if(start<end) { pos=fun(arr,start,end); quick_sort(arr,start,pos-1); quick_sort(arr,pos+1,end); }}int main(){ Edge E[MAXE]; int nume,numn; //freopen("1.txt","r",stdin);//文件輸入 printf("輸入頂數和邊數:/n"); scanf("%d%d",&numn,&nume); for(int i=0;i<nume;i++) scanf("%d%d%d",&E[i].vex1,&E[i].vex2,&E[i].weight); quick_sort(E,0,nume-1); kruskal(E,numn,nume);}

編譯運行結果:

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持武林網。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表

圖片精選

亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产精品视频网| 欧美国产日韩视频| 成人精品网站在线观看| 国产一区二区在线免费视频| 欧美精品在线观看| 丝袜亚洲另类欧美重口| 亚洲精品理论电影| 欧美日韩精品中文字幕| 亚洲一区制服诱惑| 国产美女被下药99| 中文字幕日韩av综合精品| 亚洲一区二区中文字幕| 美女国内精品自产拍在线播放| 欧美激情性做爰免费视频| 国产精品欧美激情在线播放| 亚洲片在线观看| 精品成人乱色一区二区| 国产主播精品在线| 日韩成人久久久| 欧美大片在线影院| 久热在线中文字幕色999舞| 日韩av电影手机在线| 国产一区二区三区直播精品电影| 在线看日韩欧美| 欧美精品第一页在线播放| 国产欧美精品一区二区三区-老狼| 高跟丝袜欧美一区| 国产精品扒开腿做爽爽爽的视频| 国产成人精品免费视频| 亚洲国产精品va| 国产主播精品在线| 久久久久久亚洲精品| 久久欧美在线电影| 98精品在线视频| 国产69精品久久久久久| 91美女福利视频高清| 欧美巨猛xxxx猛交黑人97人| 久久国产精彩视频| 久久黄色av网站| 91精品国产自产91精品| 久久久久久亚洲精品不卡| 国产精品视频免费在线| 精品一区二区三区三区| 国产成人av在线播放| 91探花福利精品国产自产在线| 欧洲日韩成人av| 日韩在线视频观看正片免费网站| 57pao国产成人免费| 在线视频欧美性高潮| 亚洲成人精品av| 国产精品h片在线播放| 亲爱的老师9免费观看全集电视剧| 亚洲欧美成人一区二区在线电影| www国产91| 国外日韩电影在线观看| 欧美黑人一区二区三区| 亚洲精品国产福利| 欧美激情精品久久久久久蜜臀| 国产精品永久免费视频| 91精品视频一区| 全色精品综合影院| 欧美综合一区第一页| 亚洲精品久久久久中文字幕欢迎你| 亚洲性夜色噜噜噜7777| 久久久国产精品视频| 69久久夜色精品国产69乱青草| 久久久久久有精品国产| 日本19禁啪啪免费观看www| zzijzzij亚洲日本成熟少妇| 亚洲一区二区三区xxx视频| 国产精品中文久久久久久久| 亚洲人成网站777色婷婷| 亚洲国产成人精品久久| 成人www视频在线观看| 91日本在线观看| 欧美性猛交xxxx富婆| 91精品在线国产| 久久久久北条麻妃免费看| 欧美高清一级大片| 国产日韩中文字幕| 午夜精品福利电影| 欧美成人免费小视频| 久久久国产91| 国产精品入口福利| 欧美精品激情blacked18| 亚洲专区在线视频| 日韩精品中文字幕在线| 最近2019免费中文字幕视频三| 亚洲国产又黄又爽女人高潮的| 欧美老妇交乱视频| 欧美乱大交xxxxx另类电影| 国语自产精品视频在线看抢先版图片| 亚洲图片制服诱惑| 久久夜色撩人精品| 亚洲丁香久久久| 91国产视频在线播放| 久久久亚洲天堂| 日韩精品视频在线观看免费| 国产男女猛烈无遮挡91| 久99久在线视频| 欧美激情视频一区二区| 在线播放国产一区二区三区| 国产精品国产三级国产aⅴ9色| 国产精品色悠悠| 色偷偷噜噜噜亚洲男人的天堂| 亚洲国产精品视频在线观看| 国产69久久精品成人看| 欧美激情网友自拍| 亚洲97在线观看| 麻豆成人在线看| 亚洲国产精品va在线| 亚洲欧美激情四射在线日| 久久影院资源站| 亚洲成人黄色在线| 亚洲成色777777在线观看影院| 欧美成人高清视频| 国产精品久久久| 久久亚洲一区二区三区四区五区高| 国产精品久久久久免费a∨大胸| 九九热在线精品视频| 亚洲欧美中文另类| 亚洲国产精品高清久久久| 国产日韩欧美在线视频观看| 国产精品福利观看| 欧美日韩国产色视频| 国产精品jvid在线观看蜜臀| 国产成人avxxxxx在线看| 日韩网站免费观看高清| 国产精品久久久久久影视| 久久亚洲国产成人| 日韩视频免费在线| 国产精品视频1区| 韩国日本不卡在线| 亚洲夜晚福利在线观看| 欧美成人在线免费| 精品国产拍在线观看| 丝袜美腿精品国产二区| 97在线视频免费看| 亚洲少妇中文在线| 色偷偷av亚洲男人的天堂| 久久精品久久精品亚洲人| 最近2019中文字幕在线高清| 亚洲欧美成人一区二区在线电影| 中国日韩欧美久久久久久久久| 欧美精品少妇videofree| 国产精品h片在线播放| 高清欧美性猛交xxxx黑人猛交| 成人午夜在线观看| 少妇高潮 亚洲精品| 亚洲一区制服诱惑| 国产精品久久久久久久久久东京| 亚洲xxxx在线| xxx欧美精品| 97超碰蝌蚪网人人做人人爽| 欧美成人激情图片网| 国产欧美日韩精品丝袜高跟鞋| 欧美老女人性生活| 亚洲欧美激情精品一区二区| 中文字幕亚洲自拍| 精品视频在线导航| 久久影视电视剧凤归四时歌| 久久精品在线播放| 亚洲欧美日韩中文在线| 国产精品久久久久7777婷婷|