版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進行舉報或認領(lǐng)
文檔簡介
第7章樹和二叉樹7.1樹的基本概念
7.2二叉樹概念和性質(zhì)7.3二叉樹存儲結(jié)構(gòu)7.4二叉樹的遍歷7.5二叉樹的基本運算及其實現(xiàn)7.6二叉樹的構(gòu)造7.8哈夫曼樹
7.7線索二叉樹本章小結(jié)客觀世界中許多事物存在層次關(guān)系人類社會家譜社會組織結(jié)構(gòu)圖書信息管理C文件夾1文件夾n文件1文件2文件夾11文件夾12文件11文件127.1.1樹的定義形式化定義:
樹:T={D,R}。D是包含n個節(jié)點的有窮集合(n≥0)。當(dāng)n=0時為空樹,否則關(guān)系R滿足以下條件:
有且僅有一個節(jié)點d0∈D,它對于關(guān)系R來說沒有前驅(qū)節(jié)點,節(jié)點d0稱作樹的根節(jié)點。除節(jié)點d0外,D中的每個節(jié)點對于關(guān)系R來說都有且僅有一個前驅(qū)節(jié)點。
D中每個節(jié)點對于關(guān)系R來說可以有零個或多個后繼節(jié)點。7.1樹的基本概念
遞歸定義:樹是由n(n≥0)個節(jié)點組成的有限集合(記為T)。其中:如果n=0,它是一棵空樹,這是樹的特例;如果n>0,這n個節(jié)點中存在(有僅存在)一個節(jié)點作為樹的根節(jié)點,簡稱為根節(jié)點(root),其余節(jié)點可分為m
(m>0)個互不相交的有限集T1,T2,…,Tm,其中每一棵子集本身又是一棵符合本定義的樹,稱為根root的子樹。rootT1T2Tm…7.1.2樹的表示
(1)樹形表示法。這是樹的最基本的表示,使用一棵倒置的樹表示樹結(jié)構(gòu),非常直觀和形象。下圖就是采用這種表示法。ABCDEFGJHIKLM邏輯結(jié)構(gòu)表示1
(2)文氏圖表示法。使用集合以及集合的包含關(guān)系描述樹結(jié)構(gòu)。下圖就是樹的文氏圖表示法。ABCDEFGJHIKLM邏輯結(jié)構(gòu)表示2AEFBCGJHDKLMI
(3)凹入表示法。使用線段的伸縮描述樹結(jié)構(gòu)。下圖是樹的凹入表示法。邏輯結(jié)構(gòu)表示3ABCDEFGJHIKLM
(4)括號表示法。將樹的根節(jié)點寫在括號的左邊,除根節(jié)點之外的其余節(jié)點寫在括號中并用逗號間隔來描述樹結(jié)構(gòu)。下圖是樹的括號表示法。ABCDEFGJHIKLM7.1.3樹的基本術(shù)語
1.節(jié)點的度與樹的度:樹中某個節(jié)點的子樹的個數(shù)稱為該節(jié)點的度。樹中各節(jié)點的度的最大值稱為樹的度,通常將度為m的樹稱為m次樹。2.分支節(jié)點與葉節(jié)點:度不為零的節(jié)點稱為非終端節(jié)點,又叫分支節(jié)點。度為零的節(jié)點稱為終端節(jié)點或葉節(jié)點。在分支節(jié)點中,每個節(jié)點的分支數(shù)就是該節(jié)點的度。如對于度為1的節(jié)點,其分支數(shù)為1,被稱為單分支節(jié)點;對于度為2的節(jié)點,其分支數(shù)為2,被稱為雙分支節(jié)點,其余類推。ABCDEFGJHIKLM度為3度為2
3.路徑與路徑長度:對于任意兩個節(jié)點di和dj,若樹中存在一個節(jié)點序列di,di1,di2,…,din,dj,使得序列中除di外的任一節(jié)點都是其在序列中的前一個節(jié)點的后繼,則稱該節(jié)點序列為由di到dj的一條路徑,用路徑所通過的節(jié)點序列(di,di1,di2,…,dj)表示這條路徑。
路徑長度等于路徑所通過的節(jié)點數(shù)目減1(即路徑上分支數(shù)目)。ABCDEFGJHIKLMA到K的路徑為A,D,I,K,其長度為3
4.孩子節(jié)點、雙親節(jié)點和兄弟節(jié)點:在一棵樹中,每個節(jié)點的后繼,被稱作該節(jié)點的孩子節(jié)點(或子女節(jié)點)。相應(yīng)地,該節(jié)點被稱作孩子節(jié)點的雙親節(jié)點(或父母節(jié)點)。
具有同一雙親的孩子節(jié)點互為兄弟節(jié)點。進一步推廣這些關(guān)系,可以把每個節(jié)點的所有子樹中的節(jié)點稱為該節(jié)點的子孫節(jié)點。
從樹根節(jié)點到達該節(jié)點的路徑上經(jīng)過的所有節(jié)點被稱作該節(jié)點的祖先節(jié)點。ABCDEFGJHIKLM5.節(jié)點的層次和樹的高度:樹中的每個節(jié)點都處在一定的層次上。節(jié)點的層次從樹根開始定義,根節(jié)點為第1層,它的孩子節(jié)點為第2層,以此類推,一個節(jié)點所在的層次為其雙親節(jié)點所在的層次加1。樹中節(jié)點的最大層次稱為樹的高度(或樹的深度)。
6.有序樹和無序樹:若樹中各節(jié)點的子樹是按照一定的次序從左向右安排的,且相對次序是不能隨意變換的,則稱為有序樹,否則稱為無序樹。ABCDEFGJHIKLM12347.森林:n(n>0)個互不相交的樹的集合稱為森林。森林的概念與樹的概念十分相近,因為只要把樹的根節(jié)點刪去就成了森林。反之,只要給n棵獨立的樹加上一個節(jié)點,并把這n棵樹作為該節(jié)點的子樹,則森林就變成了樹。7.1.4樹的性質(zhì)
性質(zhì)1樹中的節(jié)點數(shù)等于所有節(jié)點的度數(shù)加1。證明:根據(jù)樹的定義,在一棵樹中,除樹根節(jié)點外,每個節(jié)點有且僅有一個前驅(qū)節(jié)點。也就是說,每個節(jié)點與指向它的一個分支一一對應(yīng),所以除樹根之外的節(jié)點數(shù)等于所有節(jié)點的分支數(shù)(度數(shù)),從而可得樹中的節(jié)點數(shù)等于所有節(jié)點的度數(shù)加1。度之和=分支數(shù)分支數(shù)=n-1所以,n=度之和+1ABCDEFGJHIKLM
求解樹的節(jié)點個數(shù)問題:對于度為m的樹,在n、n0、n1、n2、…、nm中只有兩個參數(shù)未知時,一般可求出這兩個值。例如求n和n0的求解過程如下:
n=n0+n1+…+nm
度之和=n-1
度之和=n1+2n2+…+mnm
所以有:n=n1+2n2+…+mnm+1=n0+n1+…+nm
這樣:n0=n2+…+(m-1)nm+1求解方法歸納
例:一棵度為4的樹T中,若有20個度為4的節(jié)點,10個度為3的節(jié)點,1個度為2的節(jié)點,10個度為1的節(jié)點,則樹T的葉子節(jié)點個數(shù)是
。
A.41 B.82C.113 D.122注:本題為2010年全國考研題
性質(zhì)2度為m的樹中第i層上至多有mi-1個節(jié)點,這里應(yīng)有i≥1。
證明(采用數(shù)學(xué)歸納法)
對于第一層,因為樹中的第一層上只有一個節(jié)點,即整個樹的根節(jié)點,而由i=1代入mi-1,得mi-1=m1-1=1,也同樣得到只有一個節(jié)點,顯然結(jié)論成立。假設(shè)對于第(i-1)層(i>1)命題成立,即度為m的樹中第(i-1)層上至多有mi-2個節(jié)點。則根據(jù)樹的度的定義,度為m的樹中每個節(jié)點至多有m個孩子節(jié)點,所以第i層上的節(jié)點數(shù)至多為第(i-1)層上節(jié)點數(shù)的m倍,即至多為mi-2×m=mi-1個,這與命題相同,故命題成立。JIACBDHGFEKLM7.1.6樹的存儲結(jié)構(gòu)1.雙親存儲結(jié)構(gòu)
這種存儲結(jié)構(gòu)是一種順序存儲結(jié)構(gòu),用一組連續(xù)空間存儲樹的所有節(jié)點,同時在每個節(jié)點中附設(shè)一個偽指針指示其雙親節(jié)點的位置。樹的雙親存儲結(jié)構(gòu)示意圖雙親存儲結(jié)構(gòu)的類型聲明如下:typedefstruct{
ElemTypedata; //節(jié)點的值
intparent; //指向雙親的位置}PTree[MaxSize];思考題:該存儲結(jié)構(gòu)的優(yōu)缺點?2.孩子鏈存儲結(jié)構(gòu)
孩子鏈存儲結(jié)構(gòu)可按樹的度(即樹中所有節(jié)點度的最大值)設(shè)計節(jié)點的孩子節(jié)點指針域個數(shù)。以下左圖的樹對應(yīng)的孩子鏈存儲結(jié)構(gòu)如右圖所示。樹的孩子鏈存儲結(jié)構(gòu)示意圖孩子鏈存儲結(jié)構(gòu)的節(jié)點類型聲明如下:typedefstructnode{ElemTypedata; //節(jié)點的值
structnode*sons[MaxSons]; //指向孩子節(jié)點}TSonNode;其中,MaxSons為最多的孩子節(jié)點個數(shù)。思考題:有多少個空指針域?思考題:該存儲結(jié)構(gòu)的優(yōu)缺點?3.孩子兄弟鏈存儲結(jié)構(gòu)
孩子兄弟鏈存儲結(jié)構(gòu)是為每個節(jié)點設(shè)計三個域:一個數(shù)據(jù)元素域,一個該節(jié)點的第一個孩子節(jié)點指針域,一個該節(jié)點的下一個兄弟節(jié)點指針域。樹的孩子兄弟鏈存儲結(jié)構(gòu)示意圖兄弟鏈存儲結(jié)構(gòu)中節(jié)點的類型聲明如下:typedefstructtnode{ElemTypedata; //節(jié)點的值
structtnode*hp; //指向兄弟
structtnode*vp; //指向孩子節(jié)點}TSBNode;每個節(jié)點固定只有兩個指針域。思考題:該存儲結(jié)構(gòu)的優(yōu)缺點?7.2.1二叉樹概念
二叉樹是有限的節(jié)點集合。
這個集合或者是空。
或者由一個根節(jié)點和兩棵互不相交的稱為左子樹和右子樹的二叉樹組成。注意:二叉樹的定義是一種遞歸定義。7.2二叉樹概念和性質(zhì)
二叉樹的五種基本形態(tài):空樹N只含根節(jié)點L右子樹為空樹NL左右子樹均不為空樹N左子樹為空樹NRR
二叉樹是可以采用樹的邏輯結(jié)構(gòu)表示法,其四種表示法也都適用:樹形表示法文氏圖表示法凹入表示法括號表示法
在一棵二叉樹中,如果所有分支節(jié)點都有左孩子節(jié)點和右孩子節(jié)點,并且葉節(jié)點都集中在二叉樹的最下一層,這樣的二叉樹稱為滿二叉樹。
下圖所示就是一棵滿二叉樹??梢詫M二叉樹的節(jié)點進行連續(xù)編號,約定編號從樹根為1開始,按照層數(shù)從小到大、同一層從左到右的次序進行。圖中每個節(jié)點外邊的數(shù)字為對該節(jié)點的編號。
若二叉樹中最多只有最下面兩層的節(jié)點的度數(shù)可以小于2,并且最下面一層的葉節(jié)點都依次排列在該層最左邊的位置上,則這樣的二叉樹稱為完全二叉樹。
如下圖所示為一棵完全二叉樹。同樣可以對完全二叉樹中每個節(jié)點進行連續(xù)編號,編號的方法同滿二叉樹相同。圖中每個節(jié)點外邊的數(shù)字為對該節(jié)點的編號。7.2.2二叉樹性質(zhì)
性質(zhì)1非空二叉樹上葉節(jié)點數(shù)等于雙分支節(jié)點數(shù)加1。證明:設(shè)二叉樹上葉節(jié)點數(shù)為n0,單分支節(jié)點數(shù)為n1,雙分支節(jié)點數(shù)為n2,則總節(jié)點數(shù)n=n0+n1+n2。在一棵二叉樹中,所有節(jié)點的分支數(shù)(即度數(shù))應(yīng)等于單分支節(jié)點數(shù)加上雙分支節(jié)點數(shù)的2倍,即總的分支數(shù)=n1+2n2。由于二叉樹中除根節(jié)點以外,每個節(jié)點都有唯一的一個分支指向它,因此二叉樹中有:總的分支數(shù)=總節(jié)點數(shù)-1。由上述三個等式可得:n1+2n2=n0+n1+n2-1即:n0=n2+1
A
F
G
E
D
C
B求解二叉樹的節(jié)點個數(shù)問題:通常利用二叉樹的性質(zhì)1,即n0=n2+1來求解這類問題,也常利用以下關(guān)系求解:n=n0+n1+n2度之和=n-1度之和=n1+2n2所以有:
n=n1+2n2求解方法歸納
性質(zhì)2非空二叉樹上第i層上至多有2i-1個節(jié)點,這里應(yīng)有i≥1。由樹的性質(zhì)2可推出。性質(zhì)3高度為h的二叉樹至多有2h-1個節(jié)點(h≥1)。由樹的性質(zhì)3可推出。
性質(zhì)4對完全二叉樹中編號為i的節(jié)點(1≤i≤n,n≥1,n為節(jié)點數(shù))有:(1)若i≤n/2,即2i≤n,則編號為i的節(jié)點為分支節(jié)點,否則為葉子節(jié)點。(2)若n為奇數(shù),則每個分支節(jié)點都既有左孩子節(jié)點,也有右孩子節(jié)點;若n為偶數(shù),則編號最大的分支節(jié)點只有左孩子節(jié)點,沒有右孩子節(jié)點,其余分支節(jié)點都有左、右孩子節(jié)點。i/2i2i2i+1
(3)若編號為i的節(jié)點有左孩子節(jié)點,則左孩子節(jié)點的編號為2i;若編號為i的節(jié)點有右孩子節(jié)點,則右孩子節(jié)點的編號為(2i+1)。(4)除樹根節(jié)點外,若一個節(jié)點的編號為i,則它的雙親節(jié)點的編號為i/2,也就是說,當(dāng)i為偶數(shù)時,其雙親節(jié)點的編號為i/2,它是雙親節(jié)點的左孩子節(jié)點,當(dāng)i為奇數(shù)時,其雙親節(jié)點的編號為(i-1)/2,它是雙親節(jié)點的右孩子節(jié)點。i/2i2i2i+1
性質(zhì)5具有n個(n>0)節(jié)點的完全二叉樹的高度為log2n+1或log2n+1。由完全二叉樹的定義和樹的性質(zhì)3可推出。7.2.3二叉樹與樹、森林之間的轉(zhuǎn)換1.森林、樹轉(zhuǎn)換為二叉樹樹轉(zhuǎn)換為二叉樹步驟如下:(1)加線:在所有相鄰兄弟節(jié)點(森林中每棵樹的根節(jié)點可看成是兄弟節(jié)點)之間加一水平連線。(2)刪線:對每個非葉節(jié)點k,除了其最左邊的孩子節(jié)點外,刪去k與其他孩子節(jié)點的連線。(3)旋轉(zhuǎn):所有水平線段以左邊節(jié)點為軸心順時針旋轉(zhuǎn)45度。由一般的樹轉(zhuǎn)換的二叉樹的根節(jié)點的右孩子節(jié)點始終為空,原因是一般的樹的根節(jié)點不存在兄弟節(jié)點和相鄰的樹。森林轉(zhuǎn)換為二叉樹(1)轉(zhuǎn)換:將森林中的每棵樹轉(zhuǎn)換為相應(yīng)的二叉樹。(2)連接:第一棵二叉樹不動,從第二棵二叉樹開始,依次把后一棵二叉樹的根結(jié)點作為前一棵二叉樹的根結(jié)點的右孩子,直到所有的二叉樹連接在一起,即完成森林向二叉樹的轉(zhuǎn)換。(3)旋轉(zhuǎn):以根結(jié)點為軸,將整棵樹按順時針旋轉(zhuǎn)一定的角度,得到一棵層次分明的二叉樹。
(a)森林(b)森林中每棵樹對應(yīng)的二叉樹(c)森林對應(yīng)的二叉樹2023/2/339
2.二叉樹還原為樹或森林(1)連線:若某結(jié)點是其雙親的左孩子,則把該結(jié)點的右孩子、右孩子的右孩子、……,都與該結(jié)點的雙親結(jié)點用線連接起來。(2)刪線:刪除原二叉樹中所有雙親結(jié)點與右孩子結(jié)點的連線。(3)調(diào)整:調(diào)整由上兩步得到的樹或森林,使之層次分明。將一棵二叉樹還原為樹的過程
例
設(shè)森林F中有3棵樹,第一、第二和第三棵樹的節(jié)點個數(shù)分別為9、8和7,則與森林F對應(yīng)的二叉樹根節(jié)點的右子樹上的節(jié)點個數(shù)是
。
A.16 B.15 C.7 D.17思考題:
樹二叉樹,二叉樹樹為什么沒有討論樹的算法?7.3.1二叉樹的順序存儲結(jié)構(gòu)
二叉樹的順序存儲結(jié)構(gòu)中節(jié)點的存放次序是:對該樹中每個節(jié)點進行編號,其編號從小到大的順序就是節(jié)點存放在連續(xù)存儲單元的先后次序。若把二叉樹存儲到一維數(shù)組中,則該編號就是下標值加1(注意C/C++語言中數(shù)組的起始下標為0)。樹中各節(jié)點的編號與等高度的完全二叉樹中對應(yīng)位置上節(jié)點的編號相同。利用二叉樹的性質(zhì)4。7.3二叉樹存儲結(jié)構(gòu)
ABCDEFGHIJK123456789101112131415順序存儲結(jié)構(gòu)(不用下標為0的元素)ABCDEF
A
B
D#C#E######F12345678910111213142511437一般的二叉樹先用空節(jié)點補全成為完全二叉樹,然后對節(jié)點編號typedefElemTypeSqBTree[MaxSize];SqBTreebt="#ABD#C#E######F";7.3.2二叉樹的鏈式存儲結(jié)構(gòu)
在二叉樹的鏈接存儲中,節(jié)點的結(jié)構(gòu)如下:
typedefstructnode{ElemTypedata;structnode*lchild,*rchild;}BTNode;
其中,data表示值域,用于存儲對應(yīng)的數(shù)據(jù)元素,lchild和rchild分別表示左指針域和右指針域,用于分別存儲左孩子節(jié)點和右孩子節(jié)點(即左、右子樹的根節(jié)點)的存儲位置。二叉樹及其鏈式存儲結(jié)構(gòu):二叉鏈
ABCDEGFAB∧∧D∧G∧C∧E∧∧F∧b7.4.1二叉樹的基本運算概述
歸納起來,二叉樹有以下基本運算:(1)創(chuàng)建二叉樹CreateBTNode(*b,*str):根據(jù)二叉樹括號表示法的字符串*str生成對應(yīng)的鏈式存儲結(jié)構(gòu)。(2)查找節(jié)點FindNode(*b,x):在二叉樹b中尋找data域值為x的節(jié)點,并返回指向該節(jié)點的指針。(3)找孩子節(jié)點LchildNode(p)和Rchild-Node(p):分別求二叉樹中節(jié)點*p的左孩子節(jié)點和右孩子節(jié)點。7.4二叉樹的基本運算及其實現(xiàn)
(4)求高度BTNodeDepth(*b):求二叉樹b的高度。若二叉樹為空,則其高度為0;否則,其高度等于左子樹與右子樹中的最大高度加l。(5)輸出二叉樹DispBTNode(*b):以括號表示法輸出一棵二叉樹。7.4.2二叉樹的基本運算算法實現(xiàn)(1)創(chuàng)建二叉樹CreateBTNode(*b,*str)略
用ch掃描采用括號表示法表示二叉樹的字符串。分以下幾種情況:①若ch='(':則將前面剛創(chuàng)建的節(jié)點作為雙親節(jié)點進棧,并置k=1,表示其后創(chuàng)建的節(jié)點將作為這個節(jié)點的左孩子節(jié)點;②若ch=')':表示棧中節(jié)點的左右孩子節(jié)點處理完畢,退棧;③若ch=‘,’:表示其后創(chuàng)建的節(jié)點為右孩子節(jié)點,置k=2;
④其他情況:當(dāng)k=1時,表示這個節(jié)點作為棧中節(jié)點的左孩子節(jié)點;當(dāng)k=2時,表示這個節(jié)點作為棧中節(jié)點的右孩子節(jié)點。如此循環(huán)直到str處理完畢。
算法中使用一個棧St保存雙親節(jié)點,top為其棧指針,k指定其后處理的節(jié)點是雙親節(jié)點(保存在棧中)的左孩子節(jié)點(k=1)還是右孩子節(jié)點(k=2)。A(B(D(,G)),C(E,F))Ak=12BAB^^D^G^C^E^^F^DC根據(jù)括號表示法字符串構(gòu)造二叉樹voidCreateBTNode(BTNode*&b,char*str){BTNode*St[MaxSize],*p=NULL;inttop=-1,k,j=0;charch;b=NULL; //建立的二叉樹初始時為空
ch=str[j];while(ch!='\0') //str未掃描完時循環(huán)
{switch(ch){ case'(':top++;St[top]=p;k=1;break;
//為左孩子節(jié)點
case')':top--;break; case',':k=2;break;
//為孩子節(jié)點右節(jié)點default:p=(BTNode*)malloc(sizeof(BTNode)); p->data=ch;p->lchild=p->rchild=NULL; if(b==NULL) //p為二叉樹的根節(jié)點
b=p; else //已建立二叉樹根節(jié)點
{switch(k){ case1:St[top]->lchild=p;break; case2:St[top]->rchild=p;break; }}}j++;ch=str[j];}}(2)查找節(jié)點FindNode(*b,x)
采用先序遍歷遞歸算法查找值為x的節(jié)點。找到后返回其指針,否則返回NULL。算法如下:
BTNode*FindNode(BTNode*b,ElemTypex){BTNode*p;if(b==NULL)returnNULL;elseif(b->data==x)returnb;else{p=FindNode(b->lchild,x); if(p!=NULL)returnp; elsereturnFindNode(b->rchild,x);}}(3)找孩子節(jié)點LchildNode(p)和RchildNode(p)
直接返回*p節(jié)點的左孩子節(jié)點或右孩子節(jié)點的指針。算法如下:
BTNode*LchildNode(BTNode*p){returnp->lchild;}BTNode*RchildNode(BTNode*p){returnp->rchild;}(4)求高度BTNodeDepth(*b)
求二叉樹的高度的遞歸模型f()如下:
f(b)=0 b=NULLf(b)=MAX{f(b->lchild),f(b->rchild)}+1其他情況intBTNodeDepth(BTNode*b){intlchilddep,rchilddep;if(b==NULL)return(0);//空樹的高度為0else{lchilddep=BTNodeDepth(b->lchild);
//求左子樹的高度為lchilddep rchilddep=BTNodeDepth(b->rchild);
//求右子樹的高度為rchilddep return(lchilddep>rchilddep)?(lchilddep+1):(rchilddep+1));}}(5)輸出二叉樹DispBTNode(*b)
用括弧表示法輸出二叉樹。對于非空二叉樹b,先輸出其元素值,當(dāng)存在左孩子節(jié)點或右孩子節(jié)點時,輸出一個“(”符號,然后遞歸處理左子樹,輸出一個“,”符號,遞歸處理右子樹,最后輸出一個“)”符號。voidDispBTNode(BTNode*b){if(b!=NULL){printf("%c",b->data); if(b->lchild!=NULL||b->rchild!=NULL) {printf("(");
DispBTNode(b->lchild); //遞歸處理左子樹
if(b->rchild!=NULL)printf(",");
DispBTNode(b->rchild); //遞歸處理右子樹
printf(")"); }}}7.5.1二叉樹遍歷的概念
二叉樹的遍歷是指按照一定次序訪問樹中所有節(jié)點,并且每個節(jié)點僅被訪問一次的過程。它是最基本的運算,是二叉樹中所有其他運算的基礎(chǔ)。7.5二叉樹的遍歷1.先序遍歷過程先序遍歷二叉樹的過程是:
訪問根節(jié)點;先序遍歷左子樹;先序遍歷右子樹。ABCDEFGHK先序序列:ABCDEHGKF試按前序(先序)遍歷算法構(gòu)造一棵二叉樹。分析:可以參照上述前序遍歷的算法設(shè)定二叉樹各結(jié)點的值:(1)輸入根結(jié)點的值;(2)若左子樹不空,則輸入左子樹,否則輸入一個結(jié)束符;(3)若右子樹不空,則輸入右子樹,否則輸入一個結(jié)束符。對于圖所示的二叉樹,如果以@代表結(jié)束符,則按該算法輸入的順序為:ABD@@EG@@@C@F@@先序遍歷法創(chuàng)建二叉樹BTNode*CreateBinTree(){ charch; BTNode*t; //用全局變量,確定當(dāng)前要創(chuàng)建結(jié)點的值
ch=strs[i]; i++; if(ch=='@')return(NULL); else { t=(BTNode*)malloc(sizeof(BTNode));//創(chuàng)建根節(jié)點
t->data=ch; t->lchild=CreateBinTree();//創(chuàng)建左子樹
t->rchild=CreateBinTree();//創(chuàng)建右子樹
return(t);//創(chuàng)建完畢,返回節(jié)點指針
}}2.中序遍歷過程中序遍歷二叉樹的過程是:
中序遍歷左子樹;訪問根節(jié)點;中序遍歷右子樹。ABCDEFGHK中序序列:ABCDEHGKF3.后序遍歷過程后序遍歷二叉樹的過程是:
后序遍歷左子樹;后序遍歷右子樹;訪問根節(jié)點。ABCDEFGHK后序序列:ABCDEHGKF7.5.3二叉樹遍歷遞歸算法
由二叉樹的三種遍歷過程直接得到如下三種遞歸算法。先序遍歷的遞歸算法:
voidPreOrder(BTNode*b){if(b!=NULL){printf("%c",b->data);//訪問根節(jié)點
PreOrder(b->lchild);
PreOrder(b->rchild);}}ABCDEFGHK先序序列:^A形參T取值下層調(diào)用結(jié)束后返回到主調(diào)應(yīng)該執(zhí)行的語句ABCDEHGKF^B445^C4^D4555^E45^G45^H45^K45^F45遞歸算法執(zhí)行時系統(tǒng)棧的變化遞歸算法執(zhí)行過程:voidPreOrder(BTNode*b){if(b!=NULL){printf("%c",b->data);
PreOrder(b->lchild);
PreOrder(b->rchild);}}
中序遍歷的遞歸算法:
voidInOrder(BTNode*b){if(b!=NULL){InOrder(b->lchild); printf("%c",b->data);//訪問根節(jié)點
InOrder(b->rchild);}}
后序遍歷遞歸算法:
voidPostOrder(BTNode*b){if(b!=NULL){PostOrder(b->lchild);
PostOrder(b->rchild); printf("%c",b->data);//訪問根節(jié)點
}}思考題:
每種遍歷序列提供了什么信息?為什么三種遍歷都采用遞歸求解?
例7.5假設(shè)二叉樹采用二叉鏈存儲結(jié)構(gòu)存儲,試設(shè)計一個算法,計算一棵給定二叉樹的所有節(jié)點個數(shù)。
解:計算一棵二叉樹b中所有節(jié)點個數(shù)的遞歸模型f(b)如下:
f(b)=0 若b=NULL
f(b)=f(b->lchild)+f(b->rchild)+1 其他情況bb->lchildb->rchild對應(yīng)的遞歸算法如下:intNodes(BTNode*b){intnum1,num2;if(b==NULL) return0;elsereturnNodes(b->lchild)+Nodes(b->rchild)+1}提示:本題算法是基于后序遍歷的。
例7.6假設(shè)二叉樹采用二叉鏈存儲結(jié)構(gòu)存儲,試設(shè)計一個算法,計算一棵給定二叉樹的所有葉子節(jié)點個數(shù)。
解:計算一棵二叉樹b中所有葉子節(jié)點個數(shù)的遞歸模型f(b)如下:
f(b)=0 若b=NULL
f(b)=1 若*b為葉子節(jié)點
f(b)=f(b->lchild)+f(b->rchild) 其他情況intLeafNodes(BTNode*b){
intnum1,num2;
if(b==NULL)return0;elseif(b->lchild==NULL&&b->rchild==NULL)return1;else{ num1=LeafNodes(b->lchild); num2=LeafNodes(b->rchild); return(num1+num2);}}
例7.7假設(shè)二叉樹采用二叉鏈存儲結(jié)構(gòu)存儲,試設(shè)計一個算法,計算一棵給定二叉樹的所有單分支節(jié)點個數(shù)。
解:計算一棵二叉樹b中所有單分支節(jié)點個數(shù)的遞歸模型f(b)如下:
f(b)=0 若b=NULL
f(b)=f(b->lchild)+f(b->rchild)+1 若*b為單分支節(jié)點
f(b)=f(b->lchild)+f(b->rchild) 其他情況intSSonNodes(BTNode*b){
if(b==NULL)
return0;
elseif(b->lchild!=NULL&&b->rchild==NULL|| b->lchild==NULL&&b->rchild!=NULL) //為單分支節(jié)點
returnSSonNodes(b->lchild)+SSonNodes(b->rchild)+1;else //為雙分支節(jié)點或葉子節(jié)點時
returnSSonNodes(b->lchild)+SSonNodes(b->rchild);}
例7.8假設(shè)二叉樹采用二叉鏈存儲結(jié)構(gòu)存儲,試設(shè)計一個算法,計算一棵給定二叉樹中值為k的節(jié)點個數(shù)。
解:計算一棵二叉樹b中值為k的節(jié)點個數(shù)的遞歸模型f(b,k)如下:
f(b,k)=0 當(dāng)b=NULLf(b,k)=1+f(b->lchild,k)+f(b->rchild,k)當(dāng)b->data=k時
f(b,k)=f(b->lchild,k)+f(b->rchild,k) 其他情況intCountk(BTNode*b,ElemTypek){if(b==NULL)return0;elseif(b->data==k)return(1+Countk(b->lchild,k)+Countk(b->rchild,k));elsereturn(Countk(b->lchild,k)+Countk(b->rchild,k));}提示:本題算法是基于先序遍歷的。
例7.9假設(shè)二叉樹采用二叉鏈存儲結(jié)構(gòu),設(shè)計一個算法把二叉樹b復(fù)制到二叉樹t中。
解:其遞歸模型如下:
f(b,t)
t=NULL 若b=NULL
f(b,t)
復(fù)制根節(jié)點*b產(chǎn)生*t節(jié)點; 其他情況
f(b->lchild,t->lchild);f(b->rchild,t->rchild);voidCopy(BTNode*b,BTNode*&t){if(b==NULL)t=NULL;else{t=(BTNode*)malloc(sizeof(BTNode)); t->data=b->data; //復(fù)制一個根節(jié)點*t
Copy(b->lchild,t->lchild);//遞歸復(fù)制左子樹
Copy(b->rchild,t->rchild);//遞歸復(fù)制右子樹
}}
例7.10設(shè)二叉樹采用二叉鏈存儲結(jié)構(gòu),設(shè)計一個算法把二叉樹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. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 網(wǎng)絡(luò)布線合同范本
- 2024至2030年企鵝被項目投資價值分析報告
- 正規(guī)租房合同
- 貴陽市寫字樓租賃合同范本
- 保險銷售年度工作總結(jié)(7篇)
- 2024年中國男式彩棉內(nèi)衣市場調(diào)查研究報告
- 運動鞋營銷策劃書(合集4篇)
- 2025年度酒吧吧臺承包管理及夜生活服務(wù)合同3篇
- 家庭裝修合同樣本
- 2025版企業(yè)培訓(xùn)效果評估與績效提升合同3篇
- 2024年廣東省普通高中學(xué)業(yè)水平合格性地理試卷(1月份)
- 住宅樓安全性檢測鑒定方案
- 配送管理招聘面試題與參考回答2024年
- 江蘇省語文小學(xué)三年級上學(xué)期期末試題及解答參考(2024年)
- 黑龍江哈爾濱市省實驗中學(xué)2025屆數(shù)學(xué)高一上期末監(jiān)測試題含解析
- 小學(xué)一年級數(shù)學(xué)思維訓(xùn)練100題(附答案)
- 安全生產(chǎn)治本攻堅三年行動方案(一般工貿(mào)) 2024
- 2024年廣東省廣州市黃埔區(qū)中考一模語文試題及答案
- 飯?zhí)脪炜繀f(xié)議合同范本
- 2023-2024學(xué)年遼寧省重點高中沈陽市郊聯(lián)體高二上學(xué)期期末考試生物試題(解析版)
- 借款分期還款合同
評論
0/150
提交評論