數(shù)據(jù)結(jié)構(gòu)課程設(shè)計報告--dijkstra算法求最短路徑_第1頁
已閱讀1頁,還剩16頁未讀, 繼續(xù)免費(fèi)閱讀

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進(jìn)行舉報或認(rèn)領(lǐng)

文檔簡介

1、<p>  《數(shù)據(jù)結(jié)構(gòu)》課程設(shè)計</p><p>  題 目 第9題 Dijkstra算法求最短路徑 </p><p>  學(xué)生姓名 XXXX </p><p>  指導(dǎo)教師

2、 XXXX </p><p>  學(xué) 院 信息科學(xué)與工程學(xué)院 </p><p>  專業(yè)班級 XXXXXXX </p><p>  完成時間 XXXXXXX

3、 </p><p><b>  目錄</b></p><p>  問題分析與任務(wù)定義---------------------------------------------------------------------3</p><p>  1.1 課程設(shè)計題目------------------

4、-----------------------------------------------------------3</p><p>  1.2 原始數(shù)據(jù)的輸入格式--------------------------------------------------------------------3</p><p>  1.3 實現(xiàn)功能------------------------

5、-----------------------------------------------------------3</p><p>  1.4 測試用例-----------------------------------------------------------------------------------3</p><p>  1.5 問題分析--------------

6、---------------------------------------------------------------------3</p><p>  數(shù)據(jù)結(jié)構(gòu)的選擇和概要設(shè)計------------------------------------------------------------4</p><p>  2.1 數(shù)據(jù)結(jié)構(gòu)的選擇--------------------

7、------------------------------------------------------4</p><p>  2.2 概要設(shè)計-----------------------------------------------------------------------------------4</p><p>  詳細(xì)設(shè)計與編碼--------------------

8、---------------------------------------------------------6</p><p>  3.1 框架的建立---------------------------------------------------------------------------------6</p><p>  3.2 點(diǎn)結(jié)構(gòu)體的定義--------------

9、-------------------------------------------------------------7</p><p>  3.3 創(chuàng)立帶權(quán)值有向圖------------------------------------------------------------------------8</p><p>  3.4 鄰接矩陣的顯示----------------

10、-----------------------------------------------------------9</p><p>  3.5 遞歸函數(shù)的應(yīng)用---------------------------------------------------------------------------10</p><p>  3.6 Dijkstra算法實現(xiàn)最短路徑------

11、--------------------------------------------------------10</p><p>  上機(jī)調(diào)試------------------------------------------------------------------------------------11</p><p>  4.1 記錄調(diào)試過程中錯誤和問題的處理-------

12、--------------------------------------------11</p><p>  4.2 算法的時間課空間性能分析------------------------------------------------------------11</p><p>  4.3 算法的設(shè)計、調(diào)試經(jīng)驗和體會---------------------------------

13、------------------------11</p><p>  測試結(jié)果-----------------------------------------------------------------------------------12</p><p>  學(xué)習(xí)心得體會-----------------------------------------------------

14、------------------------12</p><p>  參考文獻(xiàn)-----------------------------------------------------------------------------------12</p><p>  附錄---------------------------------------------------------

15、---------------------------------------------12</p><p><b>  問題分析與任務(wù)定義</b></p><p><b>  課程設(shè)計題目:</b></p><p>  1.1題目:采用適當(dāng)?shù)拇鎯Y(jié)構(gòu)實現(xiàn)帶權(quán)有向圖的存儲,建立,輸入、顯示,以及使用Dijkstra算法,

16、尋找和輸出帶權(quán)有向圖中某個源點(diǎn)到其余各點(diǎn)的最短路徑</p><p>  1.2要求:采用適當(dāng)?shù)拇鎯Y(jié)構(gòu)實現(xiàn)帶權(quán)有向圖的存儲,建立,輸入、顯示,以及使用Dijkstra算法。</p><p>  1.3具體任務(wù):建立圖的存儲模塊,建立圖的輸出模塊,在建圖后從單源點(diǎn)開始求最短路徑,并顯示出來。</p><p><b>  原始數(shù)據(jù)的輸入格式</b>

17、</p><p>  2.1建圖:2.1.1數(shù)字</p><p>  2.2顯示:2.2.1數(shù)字+逗號+數(shù)字+回車</p><p>  2.2.2字母+回車</p><p><b>  實現(xiàn)功能</b></p><p><b>  3.1建立有向圖</b></p>

18、<p>  3.2顯示存儲的有向圖</p><p>  3.3顯示從頂點(diǎn)到其他各個頂點(diǎn)的最短路徑和是否存在路徑</p><p><b>  測試用例</b></p><p>  4.1正確數(shù)據(jù):輸入頂點(diǎn);邊值信息</p><p>  輸出結(jié)果:最短路徑是否存在,存在的情況最短路徑是多少,其次是不存在。<

19、;/p><p><b>  問題分析</b></p><p>  實現(xiàn)本程序要解決以下幾個問題:</p><p>  5.1如何存儲一個有向圖。</p><p>  5.2如何在界面中輸出該有向圖。</p><p>  5.3如何定義起始源點(diǎn)。</p><p>  5.4如何選

20、擇出最短路徑。</p><p>  5.5找到的最短路徑如何輸出。</p><p>  數(shù)據(jù)結(jié)構(gòu)的選擇和概要設(shè)計</p><p>  1.數(shù)據(jù)結(jié)構(gòu)的選擇:</p><p>  在圖的結(jié)構(gòu)中,任意兩個頂點(diǎn)之間都可能存在關(guān)系,比線性表和樹要復(fù)雜。由于不存在嚴(yán)格的前后順序,因而不能采用簡單的數(shù)組來存儲圖;另一方面,如果采用鏈表,由于圖中各頂點(diǎn)的度數(shù)

21、不盡相同,最小度數(shù)和最大度數(shù)可能相差很大,如果按最大度數(shù)的頂點(diǎn)來設(shè)計鏈表的指針域,則會浪費(fèi)很多存儲單元,反之,如果按照各個頂點(diǎn)設(shè)計不同的鏈表結(jié)點(diǎn),則會給操作帶來很大的困難。</p><p>  在此我選用鄰接矩陣的存儲結(jié)構(gòu)。采用鄰接矩陣存儲,很容易判斷圖中兩個頂點(diǎn)是否相連,也容易求出各個頂點(diǎn)的度。不過任何事情都不是完美的,采用鄰接矩陣存儲圖時,測試其邊的數(shù)目,必須檢查邊二維數(shù)組的所有元素,時間復(fù)雜度為O(n2),

22、這對于頂點(diǎn)很多而邊較少的圖(稀疏圖)是非常不合算的。</p><p>  以鄰接矩陣存儲有向圖。</p><p><b>  2.概要設(shè)計</b></p><p>  2.1對于最短路徑問題:</p><p>  最短路徑是在實際應(yīng)用中非常有用的工具,我們常見的兩種最短路徑是:</p><p> 

23、?。?)從某源點(diǎn)到其余各頂點(diǎn)之間的最短路徑。</p><p> ?。?)每一段頂點(diǎn)之間的最短路徑</p><p>  在這里我們解決第一類問題。</p><p>  2.2 Dijkstra算法用于求最短路徑:</p><p>  Dijkstra算法是按路徑長度遞增的次序逐步產(chǎn)生源點(diǎn)到其他頂點(diǎn)間的最短路徑。算法建立一個頂點(diǎn)集合S,初始時該集

24、合只有源點(diǎn)V0,然后逐步將已求得最短路徑的頂點(diǎn)加入到集合中,直到全部頂點(diǎn)都在集合S中,算法結(jié)束。</p><p>  2..3 Dijkstra算法思想</p><p>  設(shè)cost[i,j]=0,S為已經(jīng)求得最短路徑的頂點(diǎn)集合,distance[i]數(shù)組的每個元素表示當(dāng)前狀態(tài)下源點(diǎn)V0到Vi的最短路徑。算法如下: </p><p>  1) 初始化:S={V0}

25、, distance[i]=cost[0,i]。</p><p>  2) 選擇一個終點(diǎn)Vj,滿足distance[j]=MIN{ distance[i]|Vi∈V-S}。</p><p>  3)把Vj加入到S中。</p><p>  4)修改distance數(shù)組元素,修改邏輯為對于所有不在S中的頂點(diǎn)Vi.</p><p>  if(dis

26、tance[j]+cost[i,j]< distance[i]) { distance[i]= distance[j] ]+cost[i,j] }</p><p>  重復(fù)操作2)、3)、4),直到全部頂點(diǎn)加入到S中。</p><p><b>  2.4 實現(xiàn)流程</b></p><p>  在任意圖中實現(xiàn)求最短路徑問題,第一步是要能

27、成功的在內(nèi)存中輸入圖的信息,圖的信息有兩個,一是頂點(diǎn)個數(shù),二是每兩點(diǎn)之間的權(quán)值信息。當(dāng)建立圖之后,對圖進(jìn)行遍歷才能使用Dijkstra算法求出最短路徑;在完成了圖的建立之后,用Dijkstra算法的思想,從單源點(diǎn)開始,求出到各個頂點(diǎn)的最短路徑,并能夠?qū)崿F(xiàn)顯示功能。</p><p><b>  程序流程圖:</b></p><p>  (1) 輸入頂點(diǎn)個數(shù)n,邊的條數(shù),

28、初始化鄰接矩陣。</p><p>  (2)初始化所每條邊的權(quán)值與D[h]中</p><p>  (3) ① 找出v0到圖中其他各點(diǎn)的最小值</p><p> ?、?經(jīng)過改最小值的點(diǎn)到除它外其他各點(diǎn)的最小值</p><p>  ③ 直到s中的所有值全部被處理過,</p><p>  (4) 輸出各最短路徑的長度D[w]

29、</p><p>  算法的思想,從單源點(diǎn)開始,求出到各個頂點(diǎn)的最短路徑,并能夠?qū)崿F(xiàn)顯示功能。</p><p><b>  詳細(xì)設(shè)計和編碼</b></p><p><b>  3.1框架的建立</b></p><p>  typedef char VertexType;//定義圖的頂點(diǎn)為字符型 &l

30、t;/p><p>  typedef int VRType; //頂點(diǎn)關(guān)系類型</p><p>  typedef int InfoType;//該弧相關(guān)信息</p><p>  typedef struct ArcCell{</p><p>  VRType adj;//權(quán)值 </p><p>  InfoType *i

31、nfo;//弧相關(guān)信息的指針</p><p><b>  }ArcCell;</b></p><p>  typedef struct{</p><p>  VertexType vexs[MAX_VERTEX_NUM];//一維數(shù)組,存儲頂點(diǎn)</p><p>  ArcCell arcs[MAX_VERTEX_NUM]

32、[MAX_VERTEX_NUM];//鄰接矩陣 :二維數(shù)組,存儲邊和弧</p><p>  int vexnum,arcnum;//圖的當(dāng)前頂點(diǎn)數(shù)和弧數(shù) </p><p>  }MGrph;//鄰接矩陣表示的圖</p><p>  3.1.1頂點(diǎn)的定義typedef char VertexType;//定義圖的頂點(diǎn)為字符型 頂點(diǎn)的最大個數(shù)25</p>

33、<p>  3.1.2ArcCell arcs[MAX_VERTEX_NUM][MAX_VERTEX_NUM];二維數(shù)組用于存放鄰接矩陣,每個位置代表的值為圖中的權(quán)值,其余用無窮大3000表示。</p><p>  3.2點(diǎn)結(jié)構(gòu)體的定義</p><p>  /* 確定位置函數(shù) */</p><p> 

34、 int LocateVex(MGrph *G,VertexType v)</p><p><b>  {</b></p><p><b>  int j,b;</b></p><p>  for(b=0;b<G->vexnum;b++)//在所有頂點(diǎn)中尋找 </p><p>  if(

35、G->vexs[b]==v)//找到所找的頂點(diǎn)在b位置 </p><p><b>  {</b></p><p><b>  j=b;</b></p><p><b>  break;</b></p><p><b>  }//if</b></

36、p><p>  return(j);</p><p>  } //LocateVex</p><p>  3.3創(chuàng)立帶權(quán)值有向圖</p><p>  首先輸入該有向圖的頂點(diǎn)數(shù)n,然后依次輸入各個頂點(diǎn)及邊長(輸入的頂點(diǎn)的序號應(yīng)該小于頂點(diǎn)的數(shù)目)。</p><p>  /* 有向圖的建立

37、 */</p><p>  void Creat_YG(MGrph *G)</p><p><b>  {</b></p><p>  int i,k,j,n;</p><p>  VertexType v1,v2;</p><p><b>  int w=1;</b>

38、;</p><p>  printf("請輸入頂點(diǎn)個數(shù)和弧數(shù) 如括號里的方式(3,3):");//讀入頂點(diǎn)個數(shù)和弧的個數(shù)</p><p>  scanf("%d,%d",&G->vexnum,&G->arcnum);//讀出邊和弧的信息</p><p>  printf("\n"

39、); //換行輸出</p><p>  for(i=0;i<G->vexnum;i++)</p><p><b>  {</b></p><p>  printf("請輸入圖的第%d個頂點(diǎn):",w);//輸入指定的頂點(diǎn)</p><p><b>  w++;</b>&l

40、t;/p><p>  fflush(stdin);</p><p>  scanf("%c",&G->vexs[i]);</p><p>  printf("\n"); </p><p><b>  }</b></p><p>  for(i=0;

41、i<G->vexnum;i++)</p><p>  for(j=0;j<G->vexnum;j++)</p><p><b>  {</b></p><p>  G->arcs[i][j].adj=INFINITY;</p><p>  G->arcs[i][j].info=NULL

42、;</p><p><b>  }</b></p><p>  for(k=0;k<G->arcnum;k++)</p><p>  {//輸入各弧并構(gòu)造鄰接矩陣 </p><p>  printf("請輸入邊的起點(diǎn)和終點(diǎn)和權(quán)值如(v1,v2,n):");//起始點(diǎn)和終點(diǎn)和兩點(diǎn)之間對應(yīng)的權(quán)

43、值</p><p>  fflush(stdin);</p><p>  scanf("%c,%c,%d",&v1,&v2,&n);</p><p>  printf("\n"); </p><p>  i=LocateVex(G,v1);//確定v1的位置 </p>

44、<p>  j=LocateVex(G,v2);//確定v2的位置 </p><p>  G->arcs[i][j].adj=n;//邊<v1,v2>的權(quán)值賦為1,如需要權(quán)值操作則相應(yīng)修改一下即可 </p><p>  G->arcs[i][j].info=NULL;//如需要有相關(guān)信息則相對應(yīng)輸入,在這里我設(shè)置為空 </p><p

45、><b>  }//第二個for</b></p><p>  getchar();</p><p>  } //Creat_YG</p><p>  void juzhen(MGrph *G){//用矩陣來存儲并顯示出結(jié)果</p><p>  int i,j,k;</p><p>  pri

46、ntf("鄰接矩陣顯示:\n");</p><p>  printf("\t");</p><p>  for(i=0;i<G->vexnum;i++)</p><p>  printf("\t%5c",G->vexs[i]);</p><p>  for(j=0;

47、j<G->vexnum;j++)</p><p><b>  {</b></p><p>  printf("\n\n");</p><p>  printf("\t\%5c",G->vexs[j]);</p><p>  for(k=0;k<G->v

48、exnum;k++)</p><p><b>  {</b></p><p>  if(G->arcs[j][k].adj<INFINITY)</p><p>  printf("\t%5d",G->arcs[j][k].adj);</p><p>  else printf(&qu

49、ot;\t 3000"); //無權(quán)值的直接輸出最大值 </p><p><b>  }</b></p><p><b>  }</b></p><p>  3.4鄰接矩陣的顯示</p><p>  在圖的鄰接矩陣顯示中,分別利用for循環(huán)輸出了矩陣的行列標(biāo),使矩陣很明了。&

50、lt;/p><p>  void juzhen(MGrph *G){//用矩陣來存儲并顯示出結(jié)果</p><p>  int i,j,k;</p><p>  printf("鄰接矩陣顯示:\n");</p><p>  printf("\t");</p><p>  for(i=0

51、;i<G->vexnum;i++)</p><p>  printf("\t%5c",G->vexs[i]);</p><p>  for(j=0;j<G->vexnum;j++)</p><p><b>  {</b></p><p>  printf("\n

52、\n");</p><p>  printf("\t\%5c",G->vexs[j]);</p><p>  for(k=0;k<G->vexnum;k++)</p><p><b>  {</b></p><p>  if(G->arcs[j][k].adj<

53、INFINITY)</p><p>  printf("\t%5d",G->arcs[j][k].adj);</p><p>  else printf("\t 3000"); //無權(quán)值的直接輸出最大值 </p><p><b>  }</b></p><p>

54、;<b>  }</b></p><p><b>  }</b></p><p><b>  3.5遞歸函數(shù)應(yīng)用</b></p><p>  3.5.1思想是patn[]數(shù)組來表示前驅(qū)頂點(diǎn)的位置,然后遞歸輸出每個頂點(diǎn)的前驅(qū)</p><p>  void Short(MGrph

55、*G,int path[],int i,int w){</p><p>  //遞歸函數(shù)是用來輸出從源點(diǎn)出發(fā)到終點(diǎn)之前的頂點(diǎn)</p><p>  //思想是patn[]數(shù)組來表示前驅(qū)頂點(diǎn)的位置,然后遞歸輸出每個頂點(diǎn)的前驅(qū) </p><p><b>  int k;</b></p><p>  k=path[i];<

56、/p><p>  if(k!=w) Short(G,path,k,w );</p><p>  printf("%c,",G->vexs[k]);//輸出k位置的頂點(diǎn)值 </p><p><b>  }</b></p><p>  3.6 Dijkstra 算法實現(xiàn)最短路徑</p>&

57、lt;p>  設(shè)圖以鄰接矩陣cost存儲,矩陣中各元素的值為各邊的權(quán)值。頂點(diǎn)間無邊時其對應(yīng)權(quán)值用無窮大表示。從頂點(diǎn)V0到其它各頂點(diǎn)間的最短路徑的具體步驟如下:</p><p><b>  變量定義:</b></p><p>  定義整型數(shù)組S[],這是一個頂點(diǎn)集合,初始時該集合只有源點(diǎn)v0,然后逐步將以求得最短路徑的頂點(diǎn)加入到該集合中,直到全部頂點(diǎn)都在集合S中,

58、算法結(jié)束。定義兩個整型變量dis、mindis均用來標(biāo)志圖中最短的那一條路徑。</p><p><b>  初始化:</b></p><p>  初始化dist數(shù)組的值即為cost數(shù)組中存放的權(quán)值。 dist[i]=cost[v0][i] </p><p>  初始化求到每個頂點(diǎn)的最短路徑時都經(jīng)過v0頂點(diǎn)。path[i].pnode[0]=v0

59、</p><p>  初始化記錄經(jīng)過的頂點(diǎn)數(shù)都為0。 path[i].num=0; </p><p>  初始化頂點(diǎn)集合s為空,即還未開始。s[i]=0; </p><p><b>  源點(diǎn)的選擇:</b></p><p>  將v0頂點(diǎn)加入到頂點(diǎn)集合s中。 s[v0]=

60、1</p><p>  利用for循環(huán)選擇一個終點(diǎn)Vj,使其滿足V0到Vj距離最短,同時將Vj加入集合S中。</p><p>  根據(jù)j頂點(diǎn)調(diào)整當(dāng)前的最短路徑,若滿足dist[i]> dist[j]+cost[j][i],則修改dist[i]的值。同時V0到Vi的最短路徑中經(jīng)過的頂點(diǎn)數(shù)加1,即path[i].num++; 并將經(jīng)過的頂點(diǎn)存入數(shù)組pnode即path[i].pnode[

61、path[i].num]=j </p><p>  此時一趟求最短路徑完畢,將終點(diǎn)V1添加到路徑中。</p><p>  循環(huán)執(zhí)行d),e),f)操作,直到全部頂點(diǎn)加入到S中。</p><p><b>  上機(jī)調(diào)試</b></p><p>  4.1記錄調(diào)試過程中錯誤和問題的處理</p><p

62、>  1) 當(dāng)輸入格式不符合程序要求時,會出現(xiàn)循環(huán)</p><p>  2) 當(dāng)兩頂點(diǎn)間沒有路徑時,權(quán)值為無窮大,但在本程序中只能用一個具體的數(shù)字2000代替抽象的無窮大。</p><p>  3) 在程序的調(diào)試過程可暫時多加一些輸出語句以便及時查看一下中間運(yùn)行情況,并對程序規(guī)格說明作調(diào)整和改動。</p><p>  4.2算法的時間和空間性能分析</p

63、><p>  4.2.1時間復(fù)雜度</p><p>  對于n個頂點(diǎn)的有向圖,求一個頂點(diǎn)到其他頂點(diǎn)的最短路徑的時間為O(n),調(diào)整最短路徑的循環(huán)共執(zhí)行n-1次,是二層循環(huán),所以,時間復(fù)雜度是O(n2)。</p><p>  4.2.2空間復(fù)雜度</p><p>  采用鄰接矩陣存儲有向圖,應(yīng)處理每兩個頂點(diǎn)之間的關(guān)系,所以空間復(fù)雜度為O(n2)。&

64、lt;/p><p>  4.3算法設(shè)計、調(diào)試的經(jīng)驗和體會</p><p>  Dijkstra算法在上課的時候曾作為重點(diǎn)講過,所以在做查找最短路徑的算法時很流暢,但是在輸出最短路徑的時候遇到了很大的阻力。因為在定義結(jié)點(diǎn)時,使用的是結(jié)構(gòu)體數(shù)組,所以當(dāng)處理V0到每個結(jié)點(diǎn)的最短路徑時,導(dǎo)致無法具體記錄經(jīng)過的頂點(diǎn)數(shù),只能記錄源點(diǎn)、終點(diǎn)前一頂點(diǎn)以及終點(diǎn)。所以本程序在輸出最短路徑時有較大的瑕疵,還需進(jìn)一步

65、修改。</p><p><b>  測試結(jié)果</b></p><p><b>  5.1測試結(jié)果:</b></p><p><b>  注意問題:</b></p><p>  1、輸入頂點(diǎn)個數(shù):最大不超過25,請輸入羅馬數(shù)字,若輸入其他符號,無意義;</p>&l

66、t;p>  2、以“字母 字母 數(shù)字”的格式輸入圖的信息,輸入第一個字母為原點(diǎn),第二個字母為終點(diǎn),輸入“數(shù)字”為權(quán)值,最后輸入一個“字母”為頂點(diǎn)輸入。輸入完成;</p><p>  3、在輸入完成后,屏幕顯示鄰接矩陣與最短路徑。</p><p><b>  學(xué)習(xí)心得體會</b></p><p>  通過對本次課程設(shè)計的學(xué)習(xí)與交流,使我學(xué)習(xí)

67、到一部分很重要的關(guān)于編程方面思想,同時也獲得了部分學(xué)習(xí)其他學(xué)科的方法。學(xué)習(xí)重在于體會,體會這種學(xué)科給我?guī)淼乃伎?,給我?guī)碛蓽\入深的演算心得。做一次課程設(shè)計,不僅僅是為了完成某項任務(wù),而在于是否能從這次任務(wù)中總結(jié)出一些處理同類任務(wù)的方法和技巧。每次全力的付出,都會有或多或少的收獲。通過對本次課程設(shè)計涉及的問題的分析和處理,了解到學(xué)習(xí)數(shù)據(jù)結(jié)構(gòu)對編程的技巧和思想方法。以前也了解過數(shù)據(jù)結(jié)構(gòu)相關(guān)的書籍,但沒有深入的學(xué)習(xí),本次上機(jī)課程設(shè)計從選題上

68、也把學(xué)習(xí)的方法應(yīng)用其中,在編程時算法的理解和分析十分重要,首先的弄懂它的基本框架,用什么來算法來實現(xiàn),最后通過查找部分資料,修改調(diào)試,總結(jié)心得,就是一種進(jìn)步。</p><p><b>  參考文獻(xiàn)</b></p><p>  6.1:嚴(yán)蔚敏 吳偉明 編著的《數(shù)據(jù)結(jié)構(gòu)》(C語言版)</p><p>  6.2:楊曉波 主編 王 弘 王

69、聰華 胡 永 副主編的</p><p>  《數(shù)據(jù)結(jié)構(gòu)實驗指導(dǎo)》(C語言版)</p><p><b>  附件:</b></p><p>  #include<stdio.h></p><p>  #include<stdlib.h></p><p>  #defin

70、e MAX_VERTEX_NUM 25 //最大頂點(diǎn)個數(shù)</p><p>  #define INFINITY 3000//最大值</p><p>  typedef char VertexType;//定義圖的頂點(diǎn)為字符型 </p><p>  typedef int VRType; //頂點(diǎn)關(guān)系類型</p><p>  typedef

71、int InfoType;//該弧相關(guān)信息</p><p>  typedef struct ArcCell{</p><p>  VRType adj;//權(quán)值 </p><p>  InfoType *info;//弧相關(guān)信息的指針</p><p><b>  }ArcCell;</b></p><

72、;p>  typedef struct{</p><p>  VertexType vexs[MAX_VERTEX_NUM];//一維數(shù)組,存儲頂點(diǎn)</p><p>  ArcCell arcs[MAX_VERTEX_NUM][MAX_VERTEX_NUM];//鄰接矩陣 :二維數(shù)組,存儲邊和弧</p><p>  int vexnum,arcnum;//圖的

73、當(dāng)前頂點(diǎn)數(shù)和弧數(shù) </p><p>  }MGrph;//鄰接矩陣表示的圖</p><p>  /* 確定位置函數(shù) */</p><p>  int LocateVex(MGrph *G,VertexType v)</p><p><b>  {</b></p

74、><p><b>  int j,b;</b></p><p>  for(b=0;b<G->vexnum;b++)//在所有頂點(diǎn)中尋找 </p><p>  if(G->vexs[b]==v)//找到所找的頂點(diǎn)在b位置 </p><p><b>  {</b></p>

75、<p><b>  j=b;</b></p><p><b>  break;</b></p><p><b>  }//if</b></p><p>  return(j);</p><p>  } //LocateVex</p><p> 

76、 /* 有向圖的建立 */</p><p>  void Creat_YG(MGrph *G)</p><p><b>  {</b></p><p>  int i,k,j,n;</p><p>  VertexType v1,v2;</p><p&g

77、t;<b>  int w=1;</b></p><p>  printf("請輸入頂點(diǎn)個數(shù)和弧數(shù) 如括號里的方式(3,3):");//讀入頂點(diǎn)個數(shù)和弧的個數(shù)</p><p>  scanf("%d,%d",&G->vexnum,&G->arcnum);//讀出邊和弧的信息</p>&l

78、t;p>  printf("\n"); //換行輸出</p><p>  for(i=0;i<G->vexnum;i++)</p><p><b>  {</b></p><p>  printf("請輸入圖的第%d個頂點(diǎn):",w);//輸入指定的頂點(diǎn)</p><p&

79、gt;<b>  w++;</b></p><p>  fflush(stdin);</p><p>  scanf("%c",&G->vexs[i]);</p><p>  printf("\n"); </p><p><b>  }</b>&

80、lt;/p><p>  for(i=0;i<G->vexnum;i++)</p><p>  for(j=0;j<G->vexnum;j++)</p><p><b>  {</b></p><p>  G->arcs[i][j].adj=INFINITY;</p><p&g

81、t;  G->arcs[i][j].info=NULL;</p><p><b>  }</b></p><p>  for(k=0;k<G->arcnum;k++)</p><p>  {//輸入各弧并構(gòu)造鄰接矩陣 </p><p>  printf("請輸入邊的起點(diǎn)和終點(diǎn)和權(quán)值如(v1,v

82、2,n):");//起始點(diǎn)和終點(diǎn)和兩點(diǎn)之間對應(yīng)的權(quán)值</p><p>  fflush(stdin);</p><p>  scanf("%c,%c,%d",&v1,&v2,&n);</p><p>  printf("\n"); </p><p>  i=Locate

83、Vex(G,v1);//確定v1的位置 </p><p>  j=LocateVex(G,v2);//確定v2的位置 </p><p>  G->arcs[i][j].adj=n;//邊<v1,v2>的權(quán)值賦為1,如需要權(quán)值操作則相應(yīng)修改一下即可 </p><p>  G->arcs[i][j].info=NULL;//如需要有相關(guān)信息則相對

84、應(yīng)輸入,在這里我設(shè)置為空 </p><p><b>  }//第二個for</b></p><p>  getchar();</p><p>  } //Creat_YG</p><p>  void juzhen(MGrph *G){//用矩陣來存儲并顯示出結(jié)果</p><p>  int i,

85、j,k;</p><p>  printf("鄰接矩陣顯示:\n");</p><p>  printf("\t");</p><p>  for(i=0;i<G->vexnum;i++)</p><p>  printf("\t%5c",G->vexs[i]);&

86、lt;/p><p>  for(j=0;j<G->vexnum;j++)</p><p><b>  {</b></p><p>  printf("\n\n");</p><p>  printf("\t\%5c",G->vexs[j]);</p>&

87、lt;p>  for(k=0;k<G->vexnum;k++)</p><p><b>  {</b></p><p>  if(G->arcs[j][k].adj<INFINITY)</p><p>  printf("\t%5d",G->arcs[j][k].adj);</p&g

88、t;<p>  else printf("\t 3000"); //無權(quán)值的直接輸出最大值 </p><p><b>  }</b></p><p><b>  }</b></p><p><b>  }</b></p><p>

89、  void Short(MGrph *G,int path[],int i,int w){</p><p>  //遞歸函數(shù)是用來輸出從源點(diǎn)出發(fā)到終點(diǎn)之前的頂點(diǎn)</p><p>  //思想是patn[]數(shù)組來表示前驅(qū)頂點(diǎn)的位置,然后遞歸輸出每個頂點(diǎn)的前驅(qū) </p><p><b>  int k;</b></p><p&

90、gt;  k=path[i];</p><p>  if(k!=w) Short(G,path,k,w );</p><p>  printf("%c,",G->vexs[k]);//輸出k位置的頂點(diǎn)值 </p><p><b>  }</b></p><p>  void ShortestP

91、ath(MGrph *G,int w){ //Dijkstrs算法應(yīng)用</p><p>  int i,k,m,n,min;</p><p>  int final[30],path[30],D[30];//定義三個數(shù)組,final[]標(biāo)記該頂點(diǎn)是否在最短路徑上</p><p>  //path[]用來記錄頂點(diǎn)的前驅(qū)頂點(diǎn)的位置,D[]是用來記錄從源點(diǎn)到該點(diǎn)的最短路

92、徑長度 </p><p>  for(i=0;i<G->vexnum;i++)</p><p><b>  {</b></p><p>  path[i]=w;//首先把所有頂點(diǎn)的前驅(qū)頂點(diǎn)的位置賦值為w位置,也就是源點(diǎn)的位置</p><p>  final[i]=0;//這是一個標(biāo)記數(shù)組,一開始所有頂點(diǎn)位置

93、對應(yīng)的數(shù)組值全部標(biāo)記為0 </p><p>  D[i]=G->arcs[w][i].adj;//D[]是用來記錄最短路徑的長度,一開始把所有頂點(diǎn)到源點(diǎn)的的距離賦值給它 </p><p><b>  }</b></p><p>  D[w]=0;final[w]=1;path[w]=-100;//源點(diǎn)位置的D[w]為0,final[w]=

94、1標(biāo)記已經(jīng)入集合,path[w]=-100,表示該前驅(qū)為-100 </p><p>  for(m=1;m<G->vexnum;m++)</p><p><b>  {</b></p><p>  min=INFINITY;//首先賦值為機(jī)內(nèi)最大值 </p><p>  for(k=0;k<G->

95、vexnum;k++)</p><p><b>  {</b></p><p>  if(!final[k])</p><p>  if(D[k]<min)</p><p><b>  {</b></p><p><b>  i=k;</b><

96、;/p><p>  min=D[k];//最短路徑長度</p><p><b>  }</b></p><p><b>  } </b></p><p>  final[i]=1;</p><p>  for(n=0;n<G->vexnum;n++)</p&g

97、t;<p>  if(!final[n]&&(min+G->arcs[i][n].adj)<D[n])</p><p><b>  {</b></p><p>  D[n]=min+G->arcs[i][n].adj;</p><p>  path[n]=i;</p><p&

98、gt;<b>  }//if</b></p><p><b>  } //for </b></p><p>  for(m=0;m<G->vexnum;m++){</p><p>  if(final[m]==1&&m!=w)</p><p>  {printf(&qu

99、ot;\t從%c到%c的最短路徑長度為:%d\t路徑是:",G->vexs[w],G->vexs[m],D[m]);//輸出最大值時則表示兩點(diǎn)之間沒有最短路徑</p><p>  Short(G,path,m,w);</p><p>  printf("%c",G->vexs[m]);</p><p>  printf

100、("\n");</p><p><b>  }</b></p><p>  else printf("\t從%c到%c沒有不存在路徑!\n",G->vexs[w],G->vexs[m]);//判斷是否存在最短路徑</p><p><b>  }</b></p>

101、<p><b>  }</b></p><p><b>  main(){</b></p><p><b>  MGrph m;</b></p><p>  VertexType ch;</p><p><b>  int j;</b><

102、;/p><p>  Creat_YG(&m);</p><p>  juzhen(&m);</p><p>  printf("\n請輸入要開始的頂點(diǎn):");</p><p>  scanf("%c",&ch);</p><p>  j=LocateVex(&

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 眾賞文庫僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負(fù)責(zé)。
  • 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

最新文檔

評論

0/150

提交評論