數(shù)據(jù)結(jié)構(gòu)課程設(shè)計java--最小生成樹_第1頁
已閱讀1頁,還剩15頁未讀, 繼續(xù)免費閱讀

下載本文檔

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

文檔簡介

1、<p>  數(shù)據(jù)結(jié)構(gòu)(JAVA)課程設(shè)計</p><p>  題 目:____最小生成樹_______</p><p>  學(xué)生姓名:_****___________</p><p>  學(xué) 號:_____*******_______</p><p>  院 系: 計算機科學(xué)與技術(shù)學(xué)院</p><p>  專

2、業(yè)年級: ______*****___級</p><p><b>  目錄</b></p><p>  1.設(shè)計題目................................................1</p><p>  2.需求分析................................................1

3、</p><p>  1)運行環(huán)境................................................1</p><p>  2)輸入的形式和輸入值的范圍................................1</p><p>  3)輸出的形式描述........................................

4、..1</p><p>  4)功能描述................................................1</p><p>  5)測試數(shù)據(jù)...............................................1</p><p>  3.概要設(shè)計.................................

5、...............1</p><p>  1)抽象數(shù)據(jù)類型定義描述...................................1</p><p>  .2)功能模塊設(shè)計..........................................1</p><p>  3)模塊層次調(diào)用關(guān)系圖........................

6、..............2</p><p>  4.詳細(xì)設(shè)計。實現(xiàn)概要設(shè)計中定義的所有的類的定義及類中成員函數(shù),并對主要的模塊寫出偽碼算法。...........................2</p><p>  5.調(diào)試分析。包括調(diào)試過程中遇到的問題及解決的方法、算法的時間空間復(fù)雜性分析、經(jīng)驗體會。.................................6</p&

7、gt;<p>  6.用戶使用說明。詳細(xì)列出每一步的操作說明。.................7</p><p>  7. 測試結(jié)果...............................................7</p><p>  8.附錄:程序設(shè)計源代碼.....................................9</p>&

8、lt;p><b>  一、設(shè)計題目</b></p><p><b>  1).問題描述</b></p><p>  若要在 n 個城市之間建設(shè)通信網(wǎng)絡(luò),只需要架設(shè)n-1 條線路即可。如何以最低的經(jīng)濟代價建設(shè)這個通信網(wǎng),是一個網(wǎng)的最小生成樹問題。</p><p><b>  2). 基本要求</b>

9、;</p><p>  以鄰接多重表存儲無向帶權(quán)圖,利用克魯斯卡爾算法或普瑞姆算法求網(wǎng)的最小生成樹。</p><p><b>  二、需求分析</b></p><p><b>  1)運行環(huán)境</b></p><p>  軟件在JDK運行,硬件支持windows系統(tǒng)</p><p

10、>  輸入的形式和輸入值的范圍</p><p>  自動生成頂點數(shù)據(jù)在10~20之間;各個頂點之間權(quán)值在25~50之間;通過程序改動亦可生成已知頂點權(quán)值之間的最小生成樹,需將隨機生成代碼改為</p><p>  edge edge[]={new edge(0,1,16),new(0,2,18)......};</p><p>  將已知頂點、權(quán)值通過其函數(shù)輸入

11、再生成其所對應(yīng)最小生成樹。</p><p><b>  輸出的形式描述</b></p><p>  輸出隨機生成頂點個數(shù)以及各個頂點之間權(quán)值;然后輸出本次生成頂點之間構(gòu)成的最小生成樹。</p><p><b>  功能描述</b></p><p>  該程序會自動生成介于10~20個數(shù)頂點模擬各城市

12、,再隨機生成介于25~50之間數(shù)值作為權(quán)值模擬各個城市間的距離,并同時生成此次頂點、權(quán)值相對應(yīng)的最小生成樹,模擬各城市間的最小距離,最小生成樹。如有確定城市頂點及其權(quán)值,則可改動程序令其不再隨機生成頂點權(quán)值,在程序中輸入如下代碼:</p><p>  edge edge[]={new edge(0,1,16),new(0,2,18)......}</p><p>  輸入數(shù)組為edge數(shù)組

13、,edge(起點,終點,權(quán)值)。通過將隨機生成代碼改動就可以生成該城市對應(yīng)權(quán)值的最小生成樹。</p><p><b>  5)測試數(shù)據(jù)</b></p><p>  生成數(shù)據(jù)之后檢驗生成頂點數(shù)值是否介于10~20之間;檢驗各頂點間權(quán)值大小是否介于25~50間;同時檢驗其自動生成最小生成樹是否正確。</p><p><b>  三、概要設(shè)

14、計</b></p><p>  1)抽象數(shù)據(jù)類型定義描述</p><p>  定義排序類sort,將各個頂點按照其兩頂點之間權(quán)值大小排序,從大到小排序,用到堆排序算法;</p><p>  定義帶權(quán)值的邊edge,分別存在start(起點)、end(終點)、value(權(quán)值)三個變量;</p><p>  定義main類,調(diào)用so

15、rt、edge類與自身函數(shù)通過Kruskal函數(shù)實現(xiàn)最小生成樹。</p><p><b>  2)功能模塊設(shè)計</b></p><p>  主函數(shù)隨機生成10~20個頂點作為城市并同時生成任意兩頂點間25~50的權(quán)值作為兩城市距離;在界面輸出隨機生成頂點個數(shù)及任意兩頂點間權(quán)值;再調(diào)用sort函數(shù)對權(quán)進行排序,按照權(quán)值的大小有小到大排序;排序之后實現(xiàn)Kruskal函數(shù),

16、通過kruskal函數(shù)生成最小生成樹;最后輸出所生成的最小生成樹。</p><p>  3)模塊層次調(diào)用關(guān)系圖</p><p><b>  四、詳細(xì)設(shè)計</b></p><p>  實現(xiàn)概要設(shè)計中定義的所有的類的定義及類中成員函數(shù),并對主要的模塊</p><p><b>  寫出偽碼算法。</b>&

17、lt;/p><p>  定義帶權(quán)值的邊及其三個變量start(起點)、end(終點)、value(權(quán)值);定義該屬性為下邊的根據(jù)權(quán)值排序、Kruskal實現(xiàn)最小生成樹做下鋪墊;函數(shù)實現(xiàn)如下:</p><p>  package tree;</p><p>  public class sort {</p><p>  public static

18、void sift(edge a[], int root, int limit)</p><p><b>  {</b></p><p>  int i = root;</p><p>  int j = i*2+1;//j為i的左孩子</p><p>  while (j <= limit) //沿較小值孩子節(jié)點

19、向下篩選</p><p><b>  {</b></p><p>  if (j < limit && a[j].getValue() < a[j + 1].getValue())//數(shù)組元素比較</p><p><b>  {</b></p><p>  j++;//j

20、為左右孩子的較小者</p><p><b>  }</b></p><p>  if (a[j].getValue() > a[i].getValue()) //若父親節(jié)點值較大</p><p><b>  {</b></p><p>  edge e = a[i];//孩子節(jié)點中較小值上移&

21、lt;/p><p>  a[i] = a[j];</p><p><b>  a[j] = e;</b></p><p><b>  i = j;</b></p><p>  j = i * 2 + 1;//i、j向下一層</p><p><b>  }</b&g

22、t;</p><p><b>  else{</b></p><p>  break; //跳出循環(huán)</p><p><b>  }</b></p><p><b>  }</b></p><p><b>  }</b><

23、/p><p>  public static void sort(edge data[])</p><p><b>  {</b></p><p>  int length = data.length;</p><p>  for(int i = length/2-1; i>=0; i--)//創(chuàng)建最大堆</p&

24、gt;<p><b>  {</b></p><p>  sift(data, i, length-1);</p><p><b>  }</b></p><p>  for (int j = length - 1; j > 0; j--)//每趟把最大值交換到后面字,再生成堆</p>&

25、lt;p><b>  {</b></p><p>  edge e = data[0];</p><p>  data[0] = data[j];</p><p>  data[j] = e;</p><p>  sift(data, 0, j-1);</p><p><b>  

26、}</b></p><p><b>  }</b></p><p><b>  }</b></p><p>  隨機生成介于10~20之間個頂點作為各個城市,并同時生成任意兩頂點間權(quán)值,介于25~50之間;每n個頂點之間最多生成n*(n-1)條邊;生成vertexNumber-1個row(行)和row-1個co

27、lumn(列)可以防止同一個頂點生成自環(huán);</p><p><b>  函數(shù)實現(xiàn)如下:</b></p><p>  int vertexNumber=(int)((Math.random()+1)*10);</p><p>  System.out.println("隨機生成"+vertexNumber+"個頂點&

28、quot;);</p><p>  edge edges[]=new edge[vertexNumber*(vertexNumber-1)/2];</p><p>  for(int row=0, index=0; row<vertexNumber; row++)</p><p>  {//row行、column列、index數(shù)組</p><

29、;p>  for(int column=0; column<row; column++)</p><p><b>  {</b></p><p>  int x=(int)((Math.random()+1)*25);//random隨機的</p><p>  edges[index] = new edge(row, column,

30、 x);</p><p>  System.out.println("頂點"+row+"和"+column+"之間的距離為"+x);</p><p><b>  index++;</b></p><p><b>  }</b></p><p&g

31、t;<b>  }</b></p><p>  定義排序類sort,按照堆排序函數(shù)對數(shù)組edge[]按照權(quán)值大小從小到大進行排序(參照課本299頁);</p><p>  package tree;</p><p>  public class sort {</p><p>  public static void si

32、ft(edge a[], int root, int limit)</p><p><b>  {</b></p><p>  int i = root;</p><p>  int j = i*2+1;//j為i的左孩子</p><p>  while (j <= limit) //沿較小值孩子節(jié)點向下篩選<

33、;/p><p><b>  {</b></p><p>  if (j < limit && a[j].getValue() < a[j + 1].getValue())//數(shù)組元素比較</p><p><b>  {</b></p><p>  j++;//j為左右孩子的較

34、小者</p><p><b>  }</b></p><p>  if (a[j].getValue() > a[i].getValue()) //若父親節(jié)點值較大</p><p><b>  {</b></p><p>  edge e = a[i];//孩子節(jié)點中較小值上移</p&g

35、t;<p>  a[i] = a[j];</p><p><b>  a[j] = e;</b></p><p><b>  i = j;</b></p><p>  j = i * 2 + 1;//i、j向下一層</p><p><b>  }</b></

36、p><p><b>  else{</b></p><p>  break; //跳出循環(huán)</p><p><b>  }</b></p><p><b>  }</b></p><p><b>  }</b></p>

37、<p>  public static void sort(edge data[])</p><p><b>  {</b></p><p>  int length = data.length;</p><p>  for(int i = length/2-1; i>=0; i--)//創(chuàng)建最大堆</p>&l

38、t;p><b>  {</b></p><p>  sift(data, i, length-1);</p><p><b>  }</b></p><p>  for (int j = length - 1; j > 0; j--)//每趟把最大值交換到后面字,再生成堆</p><p>

39、;<b>  {</b></p><p>  edge e = data[0];</p><p>  data[0] = data[j];</p><p>  data[j] = e;</p><p>  sift(data, 0, j-1);</p><p><b>  }</b

40、></p><p><b>  }</b></p><p><b>  }</b></p><p>  Kruskal方法實現(xiàn)最小生成樹。</p><p>  Kruskal方法與Prim方法都是基于最小生成樹的MST性質(zhì):設(shè)G(V,E)是一個聯(lián)通帶權(quán)無向圖,TV是頂點集合V的一個非空真子集。

41、若(tv,v)包含于E是一條權(quán)值最小的邊,其中tv包含于TV,v包含于V-TV,則必定存在G的一棵最小生成樹T,T包含邊(tv,v)。其Kruskal算法參照課本334頁。其算法如下:</p><p>  int a[] = new int[vertexNumber];</p><p>  //初始時刻,所有頂點的連通分量編號為-1,表示所有頂點都屬于一個獨立的連通分量</p>

42、<p>  for(int i = 0; i<a.length; i++)</p><p><b>  {</b></p><p>  a[i] = -1;</p><p><b>  }</b></p><p>  edge result[] = new edge[vertex

43、Number-1];</p><p>  //該數(shù)組用于記錄最小生成樹</p><p>  int temp = 0;</p><p>  for(edge e : edges){</p><p>  int start = e.getStart();</p><p>  int end = e.getEnd();&l

44、t;/p><p>  if(a[start]==a[end] && a[end]==-1){</p><p>  a[start] = a[end] = temp;</p><p>  result[temp] = e;</p><p><b>  temp++;</b></p><p&g

45、t;<b>  }</b></p><p>  else if (a[start] != a[end]) {</p><p>  if (a[start] == -1) {</p><p>  a[start] = a[end];</p><p><b>  }</b></p><

46、;p>  else if (a[end] == -1) {</p><p>  a[end] = a[start];</p><p><b>  }</b></p><p><b>  else {</b></p><p>  int t = a[start];</p><

47、p>  for (int i = 0; i < vertexNumber; i++) {</p><p>  if (a[i] == t) {</p><p>  a[i] = a[end];</p><p><b>  }</b></p><p><b>  }</b></p&g

48、t;<p><b>  }</b></p><p>  result[temp] = e;</p><p><b>  temp++;</b></p><p><b>  }</b></p><p><b>  五、調(diào)試分析</b></

49、p><p>  包括調(diào)試過程中遇到的問題及解決的方法、算法的時間空間復(fù)雜性分析、</p><p><b>  經(jīng)驗體會。</b></p><p>  Sort排序類算法時間復(fù)雜度為O(log2n),Kruskal算法時間復(fù)雜度為O(1);</p><p>  調(diào)試過程中,Kruskal算法實現(xiàn)出現(xiàn)問題,剛開始無法實現(xiàn)該函數(shù),

50、無法生成最小生成樹;經(jīng)請教同學(xué)、查看資料、查看課本解決問題。實現(xiàn)堆排序過程無法實現(xiàn),參考課本之后解得堆排序算法實現(xiàn)過程:調(diào)用sift()方法n/2次,使得數(shù)據(jù)序列成為最大堆;對j=n-1,n-2...1,執(zhí)行下列n-1次完成排序操作:交換根end[0]和元素end[j],調(diào)用sift()方法將end[j]的前j個元素調(diào)整成最大堆。</p><p>  在編程過程中如遇難題可對其題目進行認(rèn)真分析,然后參考課本或者其

51、他資料已現(xiàn)有代碼亦或聞詢他人幫助,在自己查詢或者問詢他人過程中也是自己學(xué)習(xí)的過程,可以從中學(xué)習(xí)到很多知識。</p><p><b>  用戶使用說明</b></p><p>  程序運行后會自動跳出10~20個隨機頂點作為各個城市,同時隨機生成25~50的權(quán)值x,并生成此次所有頂點及其權(quán)值構(gòu)成的最小生成樹。</p><p><b> 

52、 測試結(jié)果</b></p><p>  1.生成0-12共13個頂點:</p><p>  2.生成最小生成樹為:</p><p>  程序隨機自動生成介于10~20之間個頂點正確運行,隨機自動生成介于25~50之間權(quán)值正確運行,使得任意兩頂點之間權(quán)值于25~50之間;經(jīng)驗證該生成樹為最小生成樹,程序運行正確。</p><p> 

53、 最小生成樹定義:設(shè)G是一個帶權(quán)連通無向圖,w(e)是邊e上的權(quán),T是G的生成樹,T中各邊的權(quán)值之和稱為生成樹T的權(quán)值或者代價(cost)。權(quán)值最小的生成樹稱之為最小生成樹(minimum cost spanning tree),簡稱最小生成樹。</p><p>  八、附錄:程序設(shè)計源代碼</p><p>  package tree;</p><p>  pub

54、lic class sort {</p><p>  public static void sift(edge a[], int root, int limit)</p><p><b>  {</b></p><p>  int i = root;</p><p>  int j = i*2+1;//j為i的左孩子&l

55、t;/p><p>  while (j <= limit) //沿較小值孩子節(jié)點向下篩選</p><p><b>  {</b></p><p>  if (j < limit && a[j].getValue() < a[j + 1].getValue())//數(shù)組元素比較</p><p>

56、;<b>  {</b></p><p>  j++;//j為左右孩子的較小者</p><p><b>  }</b></p><p>  if (a[j].getValue() > a[i].getValue()) //若父親節(jié)點值較大</p><p><b>  {</b&

57、gt;</p><p>  edge e = a[i];//孩子節(jié)點中較小值上移</p><p>  a[i] = a[j];</p><p><b>  a[j] = e;</b></p><p><b>  i = j;</b></p><p>  j = i * 2 +

58、 1;//i、j向下一層</p><p><b>  }</b></p><p><b>  else{</b></p><p>  break; //跳出循環(huán)</p><p><b>  }</b></p><p><b>  }<

59、/b></p><p><b>  }</b></p><p>  public static void sort(edge data[]){</p><p>  int length = data.length;</p><p>  for(int i = length/2-1; i>=0; i--)//創(chuàng)

60、建最大堆</p><p><b>  {</b></p><p>  sift(data, i, length-1);</p><p><b>  }</b></p><p>  for (int j = length - 1; j > 0; j--)//每趟把最大值交換到后面字,再生成堆&l

61、t;/p><p><b>  {</b></p><p>  edge e = data[0];</p><p>  data[0] = data[j];</p><p>  data[j] = e;</p><p>  sift(data, 0, j-1);</p><p>

62、<b>  }</b></p><p><b>  }</b></p><p><b>  }</b></p><p>  package tree;</p><p>  public class edge {</p><p>  private int

63、 start, end, value;//定義開始、結(jié)束、權(quán)值</p><p>  public edge(int start,int end,int value){</p><p>  this.start=start;</p><p>  this.end=end;</p><p>  this.value=value;</p>

64、;<p><b>  }</b></p><p>  public int getStart() {</p><p>  return start;</p><p><b>  }</b></p><p>  public void setStart(int start) {</

65、p><p>  this.start = start;</p><p><b>  }</b></p><p>  public int getEnd() {</p><p>  return end;</p><p><b>  }</b></p><p&g

66、t;  public void setEnd(int end) {</p><p>  this.end = end;</p><p><b>  }</b></p><p>  public int getValue() {</p><p>  return value;</p><p><

67、;b>  }</b></p><p>  public void setValue(int value) {</p><p>  this.value = value;</p><p><b>  }</b></p><p><b>  }</b></p><p

68、>  package tree;</p><p>  public class main </p><p><b>  {</b></p><p>  public static void main(String args[])</p><p><b>  {</b></p>&

69、lt;p>  int vertexNumber=(int)((Math.random()+1)*10);</p><p>  System.out.println("隨機生成"+vertexNumber+"個頂點");</p><p>  edge edges[]=new edge[vertexNumber*(vertexNumber-1)/2

70、];</p><p>  for(int row=0, index=0; row<vertexNumber; row++)</p><p>  {//row行、column列、index數(shù)組</p><p>  for(int column=0; column<row; column++)</p><p><b>  {

71、</b></p><p>  int x=(int)((Math.random()+1)*25);//random隨機的</p><p>  edges[index] = new edge(row, column, x);</p><p>  System.out.println("頂點"+row+"和"+colu

72、mn+"之間的距離為"+x);</p><p><b>  index++;</b></p><p><b>  }</b></p><p><b>  }</b></p><p>  sort.sort(edges);//對數(shù)組 edges[]中的值進行堆

73、排序</p><p>  int a[] = new int[vertexNumber];</p><p>  for(int i = 0; i<a.length; i++)</p><p>  //初始時刻,所有頂點的連通分量編號為-1,表示所有頂點都屬于一個獨立的連通分量</p><p><b>  {</b>

74、</p><p>  a[i] = -1;//a[i]的值表示第i個頂點所屬的連通分量編號</p><p><b>  }</b></p><p>  //該數(shù)組用于記錄最小生成樹</p><p>  edge result[] = new edge[vertexNumber-1];</p><p&

75、gt;  int temp = 0;</p><p>  for(edge e : edges){</p><p>  int start = e.getStart();</p><p>  int end = e.getEnd();</p><p>  if(a[start]==a[end] && a[end]==-1)//

76、只要將要加入result[]的edges的兩個頂點相等都為-1,</p><p>  //說明不和result[]中的已經(jīng)加入的聯(lián)通分量有關(guān)系,則可以直接加入result[]。</p><p><b>  {</b></p><p>  a[start] = a[end] = temp;</p><p>  result

77、[temp] = e;</p><p><b>  temp++;</b></p><p><b>  }</b></p><p>  else if (a[start] != a[end]) </p><p><b>  {</b></p><p> 

78、 if (a[start] == -1) </p><p>  //start=-1為懸空頂點,那么就讓start=end,使加入的連通分量和其連接的result[]中連通分量的標(biāo)識統(tǒng)一。</p><p><b>  {</b></p><p>  a[start] = a[end];</p><p><b>

79、  }</b></p><p>  else if (a[end] == -1) </p><p>  //end=-1為懸空頂點,那么就讓end=start,使加入的連通分量和其連接的result[]中連通分量的標(biāo)識統(tǒng)一。</p><p><b>  {</b></p><p>  a[end] = a[s

80、tart];</p><p><b>  }</b></p><p><b>  else {</b></p><p>  int t = a[start];</p><p>  for (int i = 0; i < vertexNumber; i++) </p><p&

81、gt;  //要加入的edges使得result中的兩個不同的連通分量連接起來,需將一個和另外一個進行統(tǒng)一</p><p>  //遍歷所有的頂點如果值和start相等就都等于end,則兩個連通分量進行了統(tǒng)一</p><p><b>  {</b></p><p>  if (a[i] == t) </p><p>&l

82、t;b>  {</b></p><p>  a[i] = a[end];</p><p><b>  }</b></p><p><b>  }</b></p><p><b>  }</b></p><p>  result[temp

83、] = e;//得到了result[]</p><p><b>  temp++;</b></p><p><b>  }</b></p><p>  //System.out.println("------------");</p><p>  //System.out.pri

84、ntln(Arrays.toString(a));</p><p>  if(temp == vertexNumber-1)</p><p><b>  {</b></p><p><b>  break;</b></p><p><b>  }</b></p>

85、<p><b>  }</b></p><p>  System.out.println("最小生成樹為:");</p><p>  for(edge e : result)</p><p><b>  {</b></p><p>  System.out.printl

86、n("連接頂點"+e.getStart()+"和"+e.getEnd()+"該邊的權(quán)值為"+e.getValue());</p><p><b>  }</b></p><p><b>  }</b></p><p><b>  }</b>&

溫馨提示

  • 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)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負(fù)責(zé)。
  • 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

評論

0/150

提交評論