數(shù)據(jù)結(jié)構(gòu)課程設(shè)計--樹的應(yīng)用_樹和二叉樹的轉(zhuǎn)換_第1頁
已閱讀1頁,還剩27頁未讀, 繼續(xù)免費閱讀

下載本文檔

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

文檔簡介

1、<p>  數(shù)據(jù)結(jié)構(gòu)與算法課程設(shè)計</p><p><b>  說 明 書</b></p><p>  2015 年1月 29 日</p><p><b>  一、需求分析</b></p><p>  1.設(shè)計內(nèi)容及設(shè)計要求</p><p><b>  A

2、.設(shè)計內(nèi)容:</b></p><p><b> ?。?)建立一棵樹;</b></p><p>  (2)將樹轉(zhuǎn)換成二叉樹;</p><p> ?。?)實現(xiàn)二叉樹的前序、中序、后序的遞歸和非遞歸遍歷算法。</p><p><b>  B.設(shè)計要求:</b></p><p

3、>  (1) 符合課題要求,實現(xiàn)相應(yīng)功能;</p><p>  (2) 要求界面友好美觀,操作方便易行;</p><p>  (3) 注意程序的實用性、安全性;</p><p>  2.本演示程序中,元素為單個字符,以空格表示空樹(即結(jié)點為空),以回車符作為輸入結(jié)束標(biāo)志,樹采用孩子兄弟表示法,二叉樹采用二叉鏈表表示法。在真實的運行過程中,由用戶手動輸入待創(chuàng)建樹

4、的含有空格的先根次序序列,并按回車結(jié)束,程序會將其轉(zhuǎn)化為其對應(yīng)的二叉樹,然后對二叉樹進行先序、中序、后序的遞歸及非遞歸遍歷以及層序遍歷,然后顯示轉(zhuǎn)化后二叉樹的高度和總結(jié)點數(shù),以驗證所創(chuàng)建的二叉樹是否正確,最后,銷毀創(chuàng)建的樹和二叉樹,程序結(jié)束。</p><p>  3.演示程序以用戶和計算機對話方式執(zhí)行,即在計算機終端(屏幕)上顯示“提示信息”之后,由用戶在鍵盤上輸入演示程序規(guī)定的運算命令,相應(yīng)的輸入數(shù)據(jù)和運算結(jié)果

5、顯示在其后。</p><p>  4.為了美觀,程序的輸出結(jié)果采用了分塊顯示的模式,由虛線及標(biāo)題隔開,便于用戶檢查和驗證。</p><p><b>  5.測試數(shù)據(jù)</b></p><p>  如圖,給出一棵樹,若屏幕上顯示如下信息:</p><p>  ->請按樹的先根次序輸入序列,如有空子樹,用空格填充,完成后

6、輸入回車確認(rèn)</p><p>  此時,你應(yīng)當(dāng)輸入:(↙表示回車確認(rèn))</p><p>  ABE F C DGHI J K ↙</p><p>  提示:為方便確認(rèn)輸入了幾個空格,用星號’*’表示輸入序列中的空格,則序列如下</p><p>  ABE*F**C*DGHI*J*K******↙(不是真實的輸入序列,供計算需輸入空

7、格個數(shù)時用)</p><p>  這時,建好的樹的先根和后根次序序列如下:</p><p>  樹的先根序列 A B E F C D G H I J K</p><p>  樹的后根序列 E F B C I J K H G D A</p><p>  由樹和二叉樹的轉(zhuǎn)換關(guān)系知,二叉樹的先序和

8、中序遍歷所得序列分別與樹的先根和后根遍歷所得序列相同,據(jù)此驗證轉(zhuǎn)化是否正確。二叉樹的先序和中序遍歷序列如下:</p><p>  二叉樹先序序列 A B E F C D G H I J K</p><p>  二叉樹中序序列 E F B C I J K H G D A</p><p><b>  概要設(shè)計&l

9、t;/b></p><p>  為了實現(xiàn)上述程序功能,樹采用孩子兄弟表示法,二叉樹采用二叉鏈表表示法。為此,需要兩個抽象數(shù)據(jù)類型,樹和二叉樹的抽象數(shù)據(jù)類型。</p><p>  1.樹的抽象數(shù)據(jù)類型定義</p><p>  ADT Tree{ </p><p>  數(shù)據(jù)對象D:D是具有相同特性的數(shù)據(jù)元素的集合。 &#

10、160;</p><p>  數(shù)據(jù)關(guān)系R:若D為空集,則稱為空樹; </p><p>  若D僅含有一個數(shù)據(jù)元素,則R為空集,否則R={H},H是如下二元關(guān)系: </p><p>  (1)在D中存在唯一的稱為根的數(shù)據(jù)元素root,它在關(guān)系H下無前驅(qū);  </p><p>  (2)若D-{root}≠

11、Φ,則存在D-{root}的一個劃分D1,D2,D3, ?,Dm(m>0),</p><p>  對于任意j≠k(1≤j,k≤m)有Dj∩Dk=Φ,且對任意的i(1≤i≤m),</p><p>  唯一存在數(shù)據(jù)元素xi∈Di有<root,xi>∈H;</p><p>  (3)對應(yīng)于D-{root}的劃分,H-{<root,xi&g

12、t;,?,<root,xm>}有唯一的一個劃分</p><p>  H1,H2,?,Hm(m>0),對任意j≠k(1≤j,k≤m)有Hj∩Hk=Φ,且對任意i</p><p>  (1≤i≤m),Hi是Di上的二元關(guān)系,(Di,{Hi})是一棵符合本定義的樹,</p><p>  稱為根root的子樹。 </p><p

13、><b>  基本操作P: </b></p><p>  InitTree(&T); </p><p>  操作結(jié)果:構(gòu)造空樹T。 </p><p>  DestroyTree(&T); </p><p>  初始條件:樹T存在。 </p>

14、;<p>  操作結(jié)果:銷毀樹T。 </p><p>  CreateTree(&T,definition); </p><p>  初始條件:definition給出樹T的定義。 </p><p>  操作結(jié)果:按definition構(gòu)造樹T。 </p><p>  ClearT

15、ree(&T); </p><p>  初始條件:樹T存在。 </p><p>  操作結(jié)果:將樹T清為空樹。 </p><p>  TreeEmpty(T); </p><p>  初始條件:樹T存在。 </p><p>  操作結(jié)果:若T為空樹,則返回TRU

16、E,否則返回FALSE。 </p><p>  TreeDepth(T); </p><p>  初始條件:樹T存在。 </p><p>  操作結(jié)果:返回T的深度。 </p><p><b>  Root(T); </b></p><p>  初

17、始條件:樹T存在。 </p><p>  操作結(jié)果:返回T的根。 </p><p>  Value(T,cur_e); </p><p>  初始條件:樹T存在,cur_e是T中某個結(jié)點。 </p><p>  操作結(jié)果:返回cur_e的值。 </p><p>  As

18、sign(T,cur_e,value); </p><p>  初始條件:樹T存在,cur_e是T中某個結(jié)點。 </p><p>  操作結(jié)果:結(jié)點cur_e賦值為value。 </p><p>  Parent(T,cur_e); </p><p>  初始條件:樹T存在,cur_e是T中某個結(jié)點。&

19、#160;</p><p>  操作結(jié)果:若cur_e是T的非根結(jié)點,則返回它的雙親,否則函數(shù)值為“空”。 </p><p>  LeftChild(T,cur_e); </p><p>  初始條件:樹T存在,cur_e是T中某個結(jié)點。 </p><p>  操作結(jié)果:若cur_e是T的非葉子結(jié)點,則返回它的最

20、左孩子,否則返回“空”。 </p><p>  RightSibling(T,cur_e); </p><p>  初始條件:樹T存在,cur_e是T中某個結(jié)點。 </p><p>  操作結(jié)果:若cur_e有右兄弟,則返回它的右兄弟,否則返回“空”。 </p><p>  InsertChild(&a

21、mp;T,&p,I,c); </p><p>  初始條件:樹T存在,p指向T中某個結(jié)點,1≤i≤p指結(jié)點的度+1,非空樹c與T不相交。 </p><p>  操作結(jié)果:插入c為T中p指結(jié)點的第i棵子樹。 </p><p>  DeleteChild(&T,&p,i); </p><

22、p>  初始條件:樹T存在,p指向T中某個結(jié)點,1≤i≤p指結(jié)點的度。 </p><p>  操作結(jié)果:刪除T中p所指結(jié)點的第i棵子樹。 </p><p>  TraverseTree(T,visit()); </p><p>  初始條件:樹T存在,visit是對結(jié)點操作的應(yīng)用函數(shù)。 </p><p

23、>  操作結(jié)果:按某種次序?qū)的每個結(jié)點調(diào)用函數(shù)visit( )一次且至多一次。一旦visit( )失敗,則操作失敗。 </p><p><b>  }ADT Tree</b></p><p>  2.二叉樹的抽象數(shù)據(jù)類型定義</p><p>  ADT BinaryTree{ </p><p>

24、;  數(shù)據(jù)對象D:D是具有相同特性的數(shù)據(jù)元素的集合。 </p><p><b>  數(shù)據(jù)關(guān)系R: </b></p><p>  若D=Φ,則R=Φ,稱BinaryTree為空二叉樹; </p><p>  若D≠Φ,則R={H},H是如下二元關(guān)系; </p><p> ?。?)在D中存在惟一的稱為根的數(shù)據(jù)元素ro

25、ot,它在關(guān)系H下無前驅(qū); </p><p>  (2)若D-{root}≠Φ,則存在D-{root}={D1,Dr},且D1∩Dr =Φ; </p><p> ?。?)若D1≠Φ,則D1中存在惟一的元素x1,<root,x1>∈H,且存在D1上的關(guān)系H1 ?H;若Dr≠Φ,則Dr中存在惟一的元素xr,<root,xr>∈H,且存在上的關(guān)系Hr ?H;H={&l

26、t;root,x1>,<root,xr>,H1,Hr}; </p><p>  (4)(D1,{H1})是一棵符合本定義的二叉樹,稱為根的左子樹;(Dr,{Hr})是一棵符合本定義的二叉樹,稱為根的右子樹。 </p><p><b>  基本操作: </b></p><p>  InitBiTree( &T )

27、</p><p>  操作結(jié)果:構(gòu)造空二叉樹T。</p><p>  DestroyBiTree( &T ) </p><p>  初始條件:二叉樹T已存在。 </p><p>  操作結(jié)果:銷毀二叉樹T。 </p><p>  CreateBiTree( &T, definition ) <

28、;/p><p>  初始條件:definition給出二叉樹T的定義。 </p><p>  操作結(jié)果:按definiton構(gòu)造二叉樹T。 </p><p>  ClearBiTree( &T ) </p><p>  初始條件:二叉樹T存在。 </p><p>  操作結(jié)果:將二叉樹T清為空樹。 </

29、p><p>  BiTreeEmpty( T ) </p><p>  初始條件:二叉樹T存在。 </p><p>  操作結(jié)果:若T為空二叉樹,則返回TRUE,否則返回FALSE。 </p><p>  BiTreeDepth( T ) </p><p>  初始條件:二叉樹T存在。 </p>&l

30、t;p>  操作結(jié)果:返回T的深度。 </p><p>  Root( T ) </p><p>  初始條件:二叉樹T存在。 </p><p>  操作結(jié)果:返回T的根。 </p><p>  Value( T, e ) </p><p>  初始條件:二叉樹T存在,e是T中某個結(jié)點。 </p&g

31、t;<p>  操作結(jié)果:返回e的值。 </p><p>  Assign( T, &e, value ) </p><p>  初始條件:二叉樹T存在,e是T中某個結(jié)點。 </p><p>  操作結(jié)果:結(jié)點e賦值為value。 </p><p>  Parent( T, e ) </p><

32、p>  初始條件:二叉樹T存在,e是T中某個結(jié)點。 </p><p>  操作結(jié)果:若e是T的非根結(jié)點,則返回它的雙親,否則返回“空”。 </p><p>  LeftChild( T, e ) </p><p>  初始條件:二叉樹T存在,e是T中某個結(jié)點。 </p><p>  操作結(jié)果:返回e的左孩子。若e無左孩子,則返回“

33、空”。 </p><p>  RightChild( T, e ) </p><p>  初始條件:二叉樹T存在,e是T中某個結(jié)點。 </p><p>  操作結(jié)果:返回e的右孩子。若e無右孩子,則返回“空”。 </p><p>  LeftSibling( T, e ) </p><p>  初始條件:二叉樹T

34、存在,e是T中某個結(jié)點。 </p><p>  操作結(jié)果:返回e的左兄弟。若e是T的左孩子或無左兄弟,則返回“空”。 </p><p>  RightSibling( T, e ) </p><p>  初始條件:二叉樹T存在,e是T中某個結(jié)點。 </p><p>  操作結(jié)果:返回e的右兄弟。若e是T的右孩子或無右兄弟,則返回“空”。

35、 </p><p>  InsertChild( T, p, LR, c ) </p><p>  初始條件:二叉樹T存在,p指向T中某個結(jié)點,LR為0或1,非空二叉樹c與T不相交且右子樹為空。 </p><p>  操作結(jié)果:根據(jù)LR為0或1,插入c為T中p所指結(jié)點的左或右子樹。p所指結(jié)點的原有左或右子樹則成為c的右子樹。 </p><p&

36、gt;  DeleteChild( T, p, LR ) </p><p>  初始條件:二叉樹T存在,p指向T中某個結(jié)點,LR為0或1。 </p><p>  操作結(jié)果:根據(jù)LR為0或1,刪除T中p所指結(jié)點的左或右子樹。 </p><p>  PreOrderTraverse( T, visit() ) </p><p>  初始條件

37、:二叉樹T存在,Visit是對結(jié)點操作的應(yīng)用函數(shù)。 </p><p>  操作結(jié)果:先序遍歷T,對每個結(jié)點調(diào)用函數(shù)Visit一次且僅一次。一旦visit()失敗,則操作失敗。 </p><p>  InOrderTraverse( T, visit() ) </p><p>  初始條件:二叉樹T存在,Visit是對結(jié)點操作的應(yīng)用函數(shù)。 </p>

38、<p>  操作結(jié)果:中序遍歷T,對每個結(jié)點調(diào)用函數(shù)Visit一次且僅一次。一旦visit()失敗,則操作失敗。 </p><p>  PostOrderTraverse( T, visit() ) </p><p>  初始條件:二叉樹T存在,Visit是對結(jié)點操作的應(yīng)用函數(shù)。 </p><p>  操作結(jié)果:后序遍歷T,對每個結(jié)點調(diào)用函數(shù)Visi

39、t一次且僅一次。一旦visit()失敗,則操作失敗。 </p><p>  LevelOrderTraverse( T, visit() ) </p><p>  初始條件:二叉樹T存在,Visit是對結(jié)點操作的應(yīng)用函數(shù)。 </p><p>  操作結(jié)果:層次遍歷T,對每個結(jié)點調(diào)用函數(shù)Visit一次且僅一次。一旦visit()失敗,則操作失敗。 </p&g

40、t;<p>  }ADT BinaryTree </p><p><b>  本程序包括個模塊</b></p><p><b>  【1】主程序模塊</b></p><p>  先聲明一棵樹和一棵二叉樹,然后輸入樹的含空格先根次序序列,構(gòu)建一棵樹,然后將其轉(zhuǎn)化為二叉樹,并對二叉樹進行先序、中序、后序的遞歸和非

41、遞歸遍歷以及層序遍歷,然后輸出二叉樹的高度和總結(jié)點數(shù),最后銷毀這兩棵樹。</p><p>  【2】建立樹模塊——按照樹的先根序列創(chuàng)建樹。</p><p>  【3】樹轉(zhuǎn)化二叉樹模塊——將樹轉(zhuǎn)化為二叉樹。</p><p>  【4】二叉樹的遍歷——二叉樹的先序、中序、后序的遞歸和非遞歸遍歷以及層序遍歷。</p><p>  【5】二叉樹的相關(guān)

42、信息——二叉樹的高度和總結(jié)點數(shù)。</p><p>  【6】銷毀樹和二叉樹模塊——銷毀創(chuàng)建的樹和轉(zhuǎn)換出的二叉樹。</p><p>  【7】棧和隊列的模塊——供二叉樹先序、中序、后序的非遞歸算法調(diào)用</p><p><b>  各模塊之間關(guān)系:</b></p><p><b>  詳細(xì)設(shè)計</b>&

43、lt;/p><p>  元素類型、結(jié)點類型和指針類型</p><p>  樹的結(jié)點元素類型設(shè)置為字符型,這樣既可以接收字符也可以接受數(shù)字。</p><p>  typedef char TElemType; //樹中結(jié)點元素類型</p><p>  //----------------二叉樹的二叉鏈表存儲表示-----------------

44、--- </p><p>  typedef struct BiNode{</p><p>  TElemType data; //數(shù)據(jù)域,存儲結(jié)點名稱</p><p>  struct BiNode *lchild,*rchild; //孩子結(jié)點指針 </p><p>  }BiNode,*BiTree;</p>&

45、lt;p>  二叉樹的二叉鏈表表示法示意圖:</p><p>  //-------------------樹的孩子兄弟表示法----------------------- </p><p>  typedef struct CSNode{</p><p>  TElemType data; //數(shù)據(jù)域,存儲結(jié)點名稱 </p><p>

46、;  struct CSNode *firstchild, *nextsibling; //孩子指針域和兄弟指針域 </p><p>  } CSNode, *CSTree;</p><p>  樹的孩子兄弟表示法示意圖:</p><p><b>  構(gòu)造一般樹算法:</b></p><p>  按照樹的先根次序序列構(gòu)

47、建一棵樹:</p><p>  對于這棵樹,按照需求分析第1頁的測試數(shù)據(jù),用戶應(yīng)當(dāng)輸入(↙表示回車確認(rèn))</p><p>  ABE F C DGHI J K ↙</p><p>  正確輸入所需序列后,樹的建立完成。</p><p>  Status CreateCSTree(CSTree &CT){ //按先根序列構(gòu)

48、造樹 </p><p>  //按先根次序輸入樹中結(jié)點的值(一個字符),空格字符表示空樹,構(gòu)造二叉鏈表表示樹T</p><p><b>  char ch;</b></p><p>  ch=getchar();</p><p>  if(ch==' ')</p><p><

49、b>  CT=NULL;</b></p><p><b>  else{</b></p><p>  if(!(CT=(CSNode *)malloc(sizeof(CSNode)))){</p><p>  printf("內(nèi)存分配失??!\n");</p><p>  exit(O

50、VERFLOW); </p><p><b>  }//if</b></p><p>  CT->data=ch; //生成根結(jié)點 </p><p>  CreateCSTree(CT->firstchild); //構(gòu)建左子樹 </p><p>  Create

51、CSTree(CT->nextsibling); //構(gòu)建右子樹 </p><p><b>  }//else </b></p><p>  return OK; </p><p>  }//CreateCSTree</p><p><b>  轉(zhuǎn)換為二叉樹 </b></p>

52、<p>  樹和二叉樹的轉(zhuǎn)換關(guān)鍵在于樹的二叉鏈表與其對應(yīng)的二叉樹的二叉鏈表是相同的,只是對同一個二叉鏈表的解釋不同,二叉樹的數(shù)據(jù)域存放結(jié)點名稱,左指針域指向左孩子,右指針域指向右孩子;樹的數(shù)據(jù)域存放結(jié)點名稱,左指針域指向孩子結(jié)點,右指針域指向與該結(jié)點相鄰的一個兄弟結(jié)點。當(dāng)兩者具有相同的存儲結(jié)構(gòu)時,便可以完成轉(zhuǎn)換,轉(zhuǎn)換過程的實質(zhì)是按照二叉樹的定義將二叉鏈表解釋為二叉樹的過程。</p><p>  Sta

53、tus ExchangeToBiTree(CSTree &CT,BiTree &T){ </p><p>  //將一棵用二叉鏈表表示的樹遞歸地轉(zhuǎn)換為二叉樹</p><p>  if(!CT) //樹CT為空時,二叉樹T為空</p><p><b>  T=NULL;</b></p><p><b

54、>  else{ </b></p><p>  //若樹的對應(yīng)結(jié)點不空,申請內(nèi)存空間</p><p>  if(!(T=(BiNode *)malloc(sizeof(BiNode)))){ </p><p>  printf("內(nèi)存分配失??!\n");</p><p>  exit(OVERFL

55、OW); </p><p><b>  }//if</b></p><p>  //將樹的數(shù)據(jù)域復(fù)制到二叉樹對應(yīng)的數(shù)據(jù)域</p><p>  T->data=CT->data; </p><p>  //若樹的孩子域不為空,則用樹的孩子域創(chuàng)建二叉樹的左子樹 </p>&l

56、t;p>  ExchangeToBiTree(CT->firstchild,T->lchild); </p><p>  //若樹的兄弟域不為空,則用樹的兄弟域創(chuàng)建二叉樹的右子樹 </p><p>  ExchangeToBiTree(CT->nextsibling,T->rchild); </p><p><b>  }/

57、/else </b></p><p>  }//ExchangeToBiTree</p><p><b>  二叉樹的遍歷</b></p><p>  二叉樹有先序、中序、后序、層序四種遍歷方式,而先序、中序、后序遍歷又有遞歸和非遞歸兩種算法,總共有7個遍歷算法。其中,先序、中序、后序非遞歸遍歷算法需要用到棧,層序遍歷需要使用隊列

58、,隊列和棧的相關(guān)定義和算法請參考詳細(xì)設(shè)計P21。</p><p>  二叉樹先序、中序、后序遍歷的區(qū)別僅在于訪問根結(jié)點的時機不同,而層序遍歷則是逐層從左到右訪問每一個結(jié)點。</p><p>  先序遍歷二叉樹算法的框架是:</p><p>  若二叉樹為空,則空操作;</p><p>  否則 訪問根結(jié)點 (D); 先序遍歷左子樹

59、 (L); 先序遍歷右子樹 (R)。</p><p>  中序遍歷二叉樹算法的框架是:</p><p>  若二叉樹為空,則空操作;</p><p>  否則 中序遍歷左子樹 (L); 訪問根結(jié)點 (D); 中序遍歷右子樹 (R)。</p><p>  后序遍歷二叉樹算法的框架是:</p><p>

60、  若二叉樹為空,則空操作;</p><p>  否則 后序遍歷左子樹 (L); 后序遍歷右子樹 (R); 訪問根結(jié)點 (D)。</p><p>  雖然訪問根結(jié)點的時機不同,但是先左后右的規(guī)則并沒有改變。</p><p>  所有的遍歷函數(shù)都調(diào)用元素訪問函數(shù)Visit,他們的參數(shù)列表中均含有一個函數(shù)指針變量Status(* Visit)(TElemTyp

61、e),通過主函數(shù)向他們傳遞元素訪問函數(shù)Visit的函數(shù)名,就可以實現(xiàn)按元素訪問函數(shù)Visit設(shè)定格式輸出的目的。這樣的好處在于當(dāng)輸出格式改變時,只需修改元素訪問函數(shù)Visit的輸出格式而無需更改7個遍歷函數(shù),做到一改全改。</p><p>  以下是所有遍歷算法的實現(xiàn)以及元素訪問函數(shù)Visit:</p><p>  //-----------------------元素訪問函數(shù)Visit-

62、------------------------ </p><p>  Status PrintElement(TElemType e) { //輸出元素e的值 </p><p>  printf(" %c ",e); //元素類型設(shè)定為字符型</p><p>  return OK;</p><p>  }//P

63、rintElement</p><p>  //--------------------------遞歸算法-------------------------------- </p><p>  Status PreOrderTraverse(BiTree T,Status(* Visit)(TElemType)){ //先序遍歷遞歸算法 </p><p>  /

64、/采用二叉鏈表存儲結(jié)構(gòu),Visit是對數(shù)據(jù)元素操作的應(yīng)用函數(shù)</p><p>  //先序遍歷二叉樹T的遞歸算法,對每個數(shù)據(jù)元素調(diào)用函數(shù)Visit </p><p><b>  if(T){</b></p><p>  if(Visit(T->data)) //訪問根結(jié)點</p><p>  if(PreOrder

65、Traverse(T->lchild,Visit)) //訪問左子樹</p><p>  if(PreOrderTraverse(T->rchild,Visit)) //訪問右子樹</p><p>  return OK;</p><p>  return ERROR;</p><p><b>  } //if<

66、;/b></p><p>  else return OK;</p><p>  }//PreOrderTraverse</p><p>  Status InOrderTraverse(BiTree T,Status(* Visit)(TElemType)){ //中序遍歷遞歸算法 </p><p>  //采用二叉鏈表存儲結(jié)構(gòu),V

67、isit是對數(shù)據(jù)元素操作的應(yīng)用函數(shù)</p><p>  //中序遍歷二叉樹T的遞歸算法,對每個數(shù)據(jù)元素調(diào)用函數(shù)Visit </p><p><b>  if(T){</b></p><p>  if(InOrderTraverse(T->lchild,Visit)) //訪問左子樹</p><p>  if(Vi

68、sit(T->data)) //訪問根結(jié)點</p><p>  if(InOrderTraverse(T->rchild,Visit)) //訪問右子樹</p><p>  return OK;</p><p>  return ERROR;</p><p><b>  } //if</b></p&

69、gt;<p>  else return OK;</p><p>  }//InOrderTraverse </p><p>  Status PostOrderTraverse(BiTree T,Status(* Visit)(TElemType)){ //后序遍歷遞歸算法 </p><p>  //采用二叉鏈表存儲結(jié)構(gòu),Visit是對數(shù)據(jù)元素操作

70、的應(yīng)用函數(shù)</p><p>  //后序遍歷二叉樹T的遞歸算法,對每個數(shù)據(jù)元素調(diào)用函數(shù)Visit </p><p><b>  if(T){</b></p><p>  if(PostOrderTraverse(T->lchild,Visit)) //訪問左子樹</p><p>  if(PostOrderTra

71、verse(T->rchild,Visit)) //訪問右子樹</p><p>  if(Visit(T->data)) //訪問根結(jié)點 </p><p>  return OK;</p><p>  return ERROR;</p><p><b>  }//if</b></p><

72、;p>  else return OK;</p><p>  }//PostOrderTraverse</p><p>  //-----------------------非遞歸遍歷算法---------------------------</p><p>  Status PreOrderTraverse1(BiTree T,Status(* Visit)

73、(TElemType)){ //先序遍歷非遞歸算法 </p><p>  //采用二叉鏈表存儲結(jié)構(gòu),Visit是對數(shù)據(jù)元素操作的應(yīng)用函數(shù)</p><p>  //先序遍歷二叉樹T的非遞歸算法,對每個數(shù)據(jù)元素調(diào)用函數(shù)Visit </p><p><b>  Stack S;</b></p><p>  InitStack

74、(S); //初始化棧</p><p>  BiTree p=T; //設(shè)置工作指針p并對其賦初值</p><p>  while(p||!(StackEmpty(S))){</p><p><b>  if(p){</b></p><p>  if(!Visit(p->data)) //訪問根結(jié)點 &

75、lt;/p><p>  return ERROR;</p><p>  Push(S,p); //根指針進棧 </p><p>  p=p->lchild; //遍歷左子樹</p><p><b>  }//if</b></p><p><b>  else{</

76、b></p><p>  Pop(S,p); //根指針退棧 </p><p>  p=p->rchild; //遍歷右子樹 </p><p><b>  }//else</b></p><p><b>  }//while</b></p&

77、gt;<p>  return OK;</p><p>  } //PreOrderTraverse1</p><p>  Status InOrderTraverse1(BiTree T,Status(* Visit)(TElemType)){ //中序遍歷非遞歸算法 </p><p>  //采用二叉鏈表存儲結(jié)構(gòu),Visit是對數(shù)據(jù)元素操作的應(yīng)用

78、函數(shù)</p><p>  //中序遍歷二叉樹T的非遞歸算法,對每個數(shù)據(jù)元素調(diào)用函數(shù)Visit </p><p><b>  Stack S;</b></p><p>  InitStack(S);</p><p>  BiTree p=T;</p><p>  while(p||!(StackEm

79、pty(S))){</p><p><b>  if(p){</b></p><p>  Push(S,p); //根指針進棧 </p><p>  p=p->lchild; //遍歷左子樹</p><p><b>  }//if</b></p><p>

80、;<b>  else{</b></p><p>  Pop(S,p); //根指針退棧 </p><p>  if(!Visit(p->data)) //訪問根結(jié)點 </p><p>  return ERROR;</p><p>  p=p->rchild;

81、//遍歷右子樹 </p><p><b>  }//else</b></p><p><b>  }//while</b></p><p>  return OK;</p><p>  } //InOrderTraverse1</p><p>  Status PostOrd

82、erTraverse1(BiTree T,Status(* Visit)(TElemType)){ //后序遍歷非遞歸算法 </p><p>  //采用二叉鏈表存儲結(jié)構(gòu),Visit是對數(shù)據(jù)元素操作的應(yīng)用函數(shù)</p><p>  //后序遍歷二叉樹T的非遞歸算法,對每個數(shù)據(jù)元素調(diào)用函數(shù)Visit </p><p>  BiTree p=T, q=NULL; //

83、 int n=0;</p><p><b>  Stack s;</b></p><p>  InitStack(s); </p><p>  while((p)||(!StackEmpty(s))) {</p><p><b>  while(p){</b></p><p&

84、gt;  Push(s,p);</p><p>  p=p->lchild;</p><p><b>  }//while</b></p><p><b>  q=NULL;</b></p><p>  while(!StackEmpty(s)){</p><p>  

85、GetTop(s, p);</p><p>  if((p->rchild==NULL)||(p->rchild==q)){</p><p>  if(!Visit(p->data)) //訪問根結(jié)點 </p><p>  return ERROR; </p><p>  if(p==T) return ERROR

86、;</p><p><b>  q=p;</b></p><p><b>  Pop(s,p);</b></p><p><b>  }//if</b></p><p><b>  else{</b></p><p>  p=p-&

87、gt;rchild;</p><p>  break;</p><p><b>  }//else</b></p><p><b>  }//while</b></p><p><b>  }//while</b></p><p>  retur

88、n OK;</p><p>  } //PostOrderTraverse1</p><p>  Status LevelOrderTraverse1(BiTree T,Status(* Visit)(TElemType)){ //層序遍歷非遞歸算法 </p><p>  //采用二叉鏈表存儲結(jié)構(gòu),Visit是對數(shù)據(jù)元素操作的應(yīng)用函數(shù)</p><

89、;p>  //層序遍歷二叉樹T的算法,對每個數(shù)據(jù)元素調(diào)用函數(shù)Visit </p><p><b>  Queue Q;</b></p><p>  BiTree p=T;</p><p>  if(T){ //根結(jié)點入隊列</p><p>  InitQueue(Q); //初始化隊列 &

90、lt;/p><p>  EnQueue(Q,T); </p><p>  while(!QueueEmpty(Q)){ //隊列不空 </p><p>  DeQueue(Q,p);</p><p>  if(!Visit(p->data)) //訪問根結(jié)點 </p><p>  return ERROR;&

91、lt;/p><p>  if(p->lchild)</p><p>  EnQueue(Q,p->lchild); //左孩子入隊列 </p><p>  if(p->rchild)</p><p>  EnQueue(Q,p->rchild); //右孩子入隊列 </p><p><

92、b>  }//while</b></p><p>  printf("\n");</p><p><b>  }//if</b></p><p>  return OK;</p><p>  } //LevelOrderTraverse1</p><p>&l

93、t;b>  5.二叉樹的信息</b></p><p>  int BiTreeDepth(BiTree T){ //遞歸求二叉樹高度 </p><p>  //若二叉樹T存在,返回T的深度(高度),否則返回0</p><p>  int Thigh,leftThigh,rightThigh; //分別表示二叉樹高度,左子樹高度,右子樹高度<

94、;/p><p>  if(!T) return 0; //樹高為0</p><p><b>  else{</b></p><p>  leftThigh=BiTreeDepth(T->lchild); //求左子樹高度 </p><p>  rightThigh=BiTreeDepth(T->rch

95、ild); //求右子樹高度 </p><p>  if(leftThigh>=rightThigh) //求二叉樹高度 </p><p>  Thigh=leftThigh+1;</p><p><b>  else</b></p><p>  Thigh=rightThigh+1; </p&

96、gt;<p><b>  }//else</b></p><p>  return Thigh;</p><p>  }//BiTreeDepth</p><p>  int NodeSubNum(BiTree T){ //統(tǒng)計二叉樹的結(jié)點個數(shù) </p><p>  if(!T) return 0;

97、//二叉樹為空樹,沒有結(jié)點 </p><p>  else return(NodeSubNum(T->lchild)+NodeSubNum(T->rchild)+1); </p><p>  }//NodeSubNum</p><p><b>  6.銷毀樹 </b></p><p>  void De

98、storyCSTree(CSTree &CT){</p><p>  //按照樹的定義遞歸地銷毀樹</p><p>  if(CT){ //非空樹 </p><p>  if(CT->firstchild) //孩子子樹非空</p><p>  DestoryCSTree(CT->firstchild);</p

99、><p>  if(CT->nextsibling) //兄弟子樹非空 </p><p>  DestoryCSTree(CT->nextsibling); </p><p>  free(CT); //釋放根結(jié)點</p><p>  CT=NULL; //空指針賦0 </p><p><b&g

100、t;  }//if </b></p><p>  }//DestoryTree</p><p>  void DestoryBiTree(BiTree &T){</p><p>  //按照二叉樹定義遞歸地銷毀二叉樹</p><p>  if(T){ //非空樹 </p><p>  if(T-

101、>lchild) //左子樹非空,銷毀左子樹</p><p>  DestoryBiTree(T->lchild);</p><p>  if(T->rchild) //右子樹非空,銷毀右子樹 </p><p>  DestoryBiTree(T->rchild); </p><p>  free(T); //釋

102、放根結(jié)點</p><p>  T=NULL; //空指針賦0 </p><p><b>  }//if </b></p><p>  }//DestoryTree</p><p>  void DestoryTree(CSTree &CT,BiTree &T){</p><p&g

103、t;<b>  //銷毀樹和二叉樹</b></p><p>  DestoryCSTree(CT);</p><p>  DestoryBiTree(T);</p><p>  printf("->生成的樹和二叉樹已被銷毀!"); </p><p>  }//DestoryTree</p&

104、gt;<p>  棧和隊列的定義及算法</p><p>  //-------------------棧的順序存儲表示------------------------- </p><p>  typedef BiTree SElemType; //棧的元素為二叉樹指針類型 </p><p>  typedef struct { /

105、/棧的順序存儲表示 </p><p>  SElemType *base; //棧底指針,在棧構(gòu)造之前和銷毀之后,base的值為NULL </p><p>  SElemType *top; //棧頂指針</p><p>  int stacksize;

106、 //當(dāng)前已分配的存儲空間,以元素為單位 </p><p><b>  }Stack; </b></p><p>  //------------------隊列的鏈?zhǔn)酱鎯Ρ硎?---------------------- </p><p>  typedef BiTree QElemType; //隊列元素為二叉樹指針類型</p

107、><p>  typedef struct QNode{ //鏈隊列的C語言表示 </p><p>  QElemType data; //數(shù)據(jù)域 </p><p>  struct QNode * next; //指針域 </p><p>  }QNode,* QueuePtr;</

108、p><p>  typedef struct{</p><p>  QueuePtr front; //隊頭指針 </p><p>  QueuePtr rear; //隊尾指針 </p><p><b>  }Queue; </b></p><p>  //--------------棧的相關(guān)

109、函數(shù)(供非遞歸后序遍歷使用)-------------------</p><p>  Status InitStack(Stack &S){//構(gòu)造一個空的順序棧 </p><p>  if(!(S.base=(SElemType *)malloc(STACK_INIT_SIZE*sizeof(SElemType)))){</p><p>  printf

110、("內(nèi)存分配失??!\n");</p><p>  exit(OVERFLOW);</p><p><b>  } </b></p><p>  S.top=S.base;</p><p>  S.stacksize=STACK_INIT_SIZE;</p><p>  

111、return OK; </p><p>  }//InitStack</p><p>  Status DestoryStack(Stack &S){//釋放順序棧S所占內(nèi)存空間 </p><p>  free(S.base);</p><p>  S.base=NULL;</p><p>  S.top=NU

112、LL; </p><p>  return OK; </p><p>  }//DestoryStack</p><p>  Status StackEmpty(Stack S){//判斷順序棧S是否為空棧,是返回1,否返回0 </p><p>  return S.top==S.base; </p><p>  }/

113、/StackIsEmpty</p><p>  Status Push(Stack &S,SElemType e){ //入棧 </p><p>  if(S.top-S.base >= S.stacksize){</p><p>  if(!(S.base=(SElemType *)realloc(S.base,(STACK_INIT_SIZE +

114、 STACKINCREMENT)*sizeof(SElemType)))){</p><p>  printf("內(nèi)存分配失??!\n");</p><p>  exit(OVERFLOW);</p><p><b>  } </b></p><p>  S.top = S.base + S.s

115、tacksize;</p><p>  S.stacksize +=STACKINCREMENT;</p><p><b>  }</b></p><p>  *S.top++=e; </p><p><b>  }//Push</b></p><p>  Status P

116、op(Stack &S,SElemType &e){ //出棧 </p><p>  if(StackEmpty(S)) </p><p>  return ERROR; </p><p>  e=*--S.top; </p><p>  return OK; </p><p><b>  

117、}//Pop</b></p><p>  Status GetTop(Stack S,SElemType &e){</p><p>  //若棧不空,用e返回順序棧S棧頂元素的值,并返回OK,否則返回ERRROR </p><p>  if(StackEmpty(S)) </p><p>  return ERROR;&

118、lt;/p><p>  e = *(S.top-1); </p><p>  return OK; </p><p><b>  }//GetTop</b></p><p>  //-----------------隊列的相關(guān)函數(shù)(供非遞歸層序遍歷使用)---------------</p><p>

119、;  QueuePtr MallocQNode(){//為鏈隊列結(jié)點申請內(nèi)存的函數(shù)</p><p>  QueuePtr p; //工作指針p </p><p>  if(!(p=(QueuePtr)malloc(sizeof(QNode)))){ //申請結(jié)點的內(nèi)存空間,若失敗則提示并退出程序</p><p>  printf("內(nèi)存分配失敗,程序即

120、將退出!\n");</p><p>  exit(OVERFLOW);</p><p><b>  }</b></p><p><b>  return p;</b></p><p>  } //MallocQNode </p><p>  Status InitQ

121、ueue(Queue &Q) //初始化鏈隊列 </p><p>  { //構(gòu)建一個空隊列 Q</p><p>  Q.front=Q.rear=MallocQNode(); //申請頭結(jié)點的內(nèi)存空間,并使隊頭和隊尾指針同時指向它 </p><p>  Q.front->next=NULL; </p><p>  

122、Q.front->data=0; //將隊長設(shè)為0 </p><p>  return OK;</p><p>  }//InitQueue</p><p>  Status DestoryQueue(Queue &Q){//銷毀隊列Q</p><p>  while(Q.front){ //從頭結(jié)點開始向后逐個釋放結(jié)

123、點內(nèi)存空間 </p><p>  Q.rear=Q.front->next;</p><p>  free(Q.front);</p><p>  Q.front=Q.rear; </p><p><b>  }//while</b></p><p>  printf("鏈隊列已成

124、功銷毀!\n");</p><p>  return OK;</p><p>  }//DestoryQueue</p><p>  Status QueueEmpty(Queue Q){ //若Q為空隊列,則返回OK;否則返回ERROR</p><p>  if(Q.rear==Q.front) //隊列為空的標(biāo)志 </

125、p><p>  return OK; </p><p>  return ERROR; </p><p>  }//QueueEmpty</p><p>  Status EnQueue(Queue &Q,QElemType e){</p><p>  //插入元素e為Q的新的隊尾元素</p><

126、;p>  QueuePtr p=MallocQNode(); </p><p>  p->data=e;</p><p>  p->next=NULL;</p><p>  Q.rear->next=p;</p><p><b>  Q.rear=p;</b></p><p&g

127、t;  Q.front->data++; //隊長+1 </p><p>  return OK; </p><p>  }//EnQueue</p><p>  Status DeQueue(Queue &Q,QElemType &e)</p><p>  { //若隊列不空,則刪除Q的隊頭元素,用e返回其值,并

128、返回OK,否則返回ERROR</p><p>  if(QueueEmpty(Q)) </p><p>  return ERROR;</p><p>  QueuePtr p= Q.front->next;</p><p>  e = p->data;</p><p>  Q.front->next

129、 = p->next;</p><p>  if(Q.rear==p)</p><p>  Q.rear=Q.front;</p><p><b>  free(p);</b></p><p>  Q.front->data--; //隊長-1 </p><p>  return O

130、K; </p><p>  }//DeQueue</p><p><b>  主函數(shù)</b></p><p>  int main(int argc,char *argv[]){</p><p>  printf("------------------------ 樹的應(yīng)用 ----------------

131、---------\n");</p><p>  BiTree T=NULL; //聲明一棵二叉樹</p><p>  CSTree CT=NULL; //聲明一棵普通樹</p><p>  printf(" -----------------樹的建立---------------- \n"

溫馨提示

  • 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

提交評論