版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進(jìn)行舉報或認(rèn)領(lǐng)
文檔簡介
1樹與二叉樹2樹和森林的概念兩種樹:自由樹與有根樹。
自由樹: 一棵自由樹Tf
可定義為一個二元組Tf
=(V,E)
其中V={v1,...,vn}
是由n(n>0)個元素組成的有限非空集合,稱為頂點(diǎn)集合。E={(vi,vj)|vi,vj
V,1≤i,j≤n}
是n-1個序?qū)Φ募希Q為邊集合,E
中的元素(vi,vj)稱為邊或分支。3自由樹有根樹: 一棵有根樹T,簡稱為樹,它是n(n≥0)
個結(jié)點(diǎn)的有限集合。當(dāng)n=0時,T稱為空樹;否則,T
是非空樹,記作
4r是一個特定的稱為根(root)的結(jié)點(diǎn),它只有直接后繼,但沒有直接前驅(qū);根以外的其他結(jié)點(diǎn)劃分為m(m
0)個互不相交的有限集合T1,T2,…,Tm,每個集合又是一棵樹,并且稱之為根的子樹。每棵子樹的根結(jié)點(diǎn)有且僅有一個直接前驅(qū),但可以有0個或多個直接后繼。5樹的基本術(shù)語子女:若結(jié)點(diǎn)的子樹非空,結(jié)點(diǎn)子樹的根即為該結(jié)點(diǎn)的子女。雙親:若結(jié)點(diǎn)有子女,該結(jié)點(diǎn)是子女雙親。1層2層4層3層depth=4DACBIJHGFEMLK6兄弟:同一結(jié)點(diǎn)的子女互稱為兄弟。度:結(jié)點(diǎn)的子女個數(shù)即為該結(jié)點(diǎn)的度;樹中各個結(jié)點(diǎn)的度的最大值稱為樹的度。分支結(jié)點(diǎn):度不為0的結(jié)點(diǎn)即為分支結(jié)點(diǎn),亦稱為非終端結(jié)點(diǎn)。葉結(jié)點(diǎn):度為0的結(jié)點(diǎn)即為葉結(jié)點(diǎn),亦稱為終端結(jié)點(diǎn)。祖先:某結(jié)點(diǎn)到根結(jié)點(diǎn)的路徑上的各個結(jié)點(diǎn)都是該結(jié)點(diǎn)的祖先。子孫:某結(jié)點(diǎn)的所有下屬結(jié)點(diǎn),都是該結(jié)點(diǎn)的子孫。7結(jié)點(diǎn)的層次:規(guī)定根結(jié)點(diǎn)在第一層,其子女結(jié)點(diǎn)的層次等于它的層次加一。以下類推。深度:結(jié)點(diǎn)的深度即為結(jié)點(diǎn)的層次;離根最遠(yuǎn)結(jié)點(diǎn)的層次即為樹的深度。1層2層4層3層depth=4DACBIJHGFEMLK8高度:規(guī)定葉結(jié)點(diǎn)的高度為1,其雙親結(jié)點(diǎn)的高度等于它的高度加一。有序樹:樹中結(jié)點(diǎn)的各棵子樹T0,T1,…是有次序的,即為有序樹。無序樹:樹中結(jié)點(diǎn)的各棵子樹之間的次序是不重要的,可以互相交換位置。森林:森林是m(m≥0)棵樹的集合。
9樹的抽象數(shù)據(jù)類型template<classT>classTree{//對象:樹是由n(≥0)個結(jié)點(diǎn)組成的有限集合。在//類界面中的position是樹中結(jié)點(diǎn)的地址。在順序//存儲方式下是下標(biāo)型,在鏈表存儲方式下是指針//型。T是樹結(jié)點(diǎn)中存放數(shù)據(jù)的類型,要求所有結(jié)//點(diǎn)的數(shù)據(jù)類型都是一致的。public:Tree();
~Tree();
10BuildRoot(constT&value);
//建立樹的根結(jié)點(diǎn)
positionFirstChild(positionp);
//返回
p第一個子女地址,無子女返回0 positionNextSibling(positionp);
//返回p下一兄弟地址,若無下一兄弟返回0 positionParent(positionp);
//返回
p雙親結(jié)點(diǎn)地址,若
p為根返回0TGetData(positionp);
//返回結(jié)點(diǎn)
p中存放的值
boolInsertChild(positionp,T&value);
//在結(jié)點(diǎn)
p下插入值為
value的新子女,若插
//入失敗,函數(shù)返回false,否則返回true11
boolDeleteChild(positionp,inti);
//刪除結(jié)點(diǎn)p
的第
i
個子女及其全部子孫結(jié)
//點(diǎn),若刪除失敗,則返回false,否則返回true
voidDeleteSubTree(positiont);
//刪除以t
為根結(jié)點(diǎn)的子樹
boolIsEmpty();
//判樹空否,若空則返回true,否則返回false
voidTraversal(positionp);
//遍歷以
p
為根的子樹};
12二叉樹的五種不同形態(tài)LLRR二叉樹(BinaryTree)二叉樹的定義
一棵二叉樹是結(jié)點(diǎn)的一個有限集合,該集合或者為空,或者是由一個根結(jié)點(diǎn)加上兩棵分別稱為左子樹和右子樹的、互不相交的二叉樹組成。13二叉樹的性質(zhì)性質(zhì)1
若二叉樹結(jié)點(diǎn)的層次從1開始,則在二叉樹的第i層最多有
2i-1
個結(jié)點(diǎn)。(i≥1)
[證明用數(shù)學(xué)歸納法]性質(zhì)2
深度為k
的二叉樹最少有k
個結(jié)點(diǎn),最多有2k-1個結(jié)點(diǎn)。(k≥1)
因?yàn)槊恳粚幼钌僖?個結(jié)點(diǎn),因此,最少結(jié)點(diǎn)數(shù)為k。最多結(jié)點(diǎn)個數(shù)借助性質(zhì)1:用求等比級數(shù)前k項和的公式20+21+22+…+2k-1=2k-114性質(zhì)3
對任何一棵二叉樹,如果其葉結(jié)點(diǎn)有n0個,度為2的非葉結(jié)點(diǎn)有n2個,則有n0=n2+1
若設(shè)度為1的結(jié)點(diǎn)有n1個,總結(jié)點(diǎn)數(shù)為n, 總邊數(shù)為e,則根據(jù)二叉樹的定義,
n=n0+n1+n2
e
=2n2+n1=n-1
因此,有2n2+n1=n0+n1+n2-1
n2=n0-1n0=n2+115定義1
滿二叉樹(FullBinaryTree)
定義2
完全二叉樹(CompleteBinaryTree)─若設(shè)二叉樹的深度為k,則共有k層。除第k層外,其它各層(1~k-1)的結(jié)點(diǎn)數(shù)都達(dá)到最大個數(shù),第k層從右向左連續(xù)缺若干結(jié)點(diǎn),這就是完全二叉樹。16性質(zhì)4
具有n(n≥0)個結(jié)點(diǎn)的完全二叉樹的深度為log2(n+1)
設(shè)完全二叉樹的深度為k,則有
2k-1-1<n
≤2k-1
變形2k-1<n+1≤2k
取對數(shù)
k-1<log2(n+1)≤k
有
log2(n+1)=k上面k-1層結(jié)點(diǎn)數(shù)包括第k層的最大結(jié)點(diǎn)數(shù)23-124-117性質(zhì)5
如將一棵有n個結(jié)點(diǎn)的完全二叉樹自頂向下,同一層自左向右連續(xù)給結(jié)點(diǎn)編號1,2,…,n,則有以下關(guān)系:
若i=1,則i無雙親若i>1,則i的雙親為i/2若2*i<=n,則i的左子女為
2*i, 若2*i+1<=n,則i的右子女為2*i+1若i為奇數(shù),且i!=1,
則其左兄弟為i-1,若若
i為偶數(shù),且i!=n,
則其右兄弟為i+11234856791018二叉樹的抽象數(shù)據(jù)類型template<classT>classBinaryTree{//對象:結(jié)點(diǎn)的有限集合,二叉樹是有序樹public:
BinaryTree(); //構(gòu)造函數(shù)
BinaryTree(BinTreeNode<T>*lch,
BinTreeNode<T>*rch,Titem);
//構(gòu)造函數(shù),以item為根,lch和rch為左、右子
//樹構(gòu)造一棵二叉樹
intDepth(); //求樹深度
intSize(); //求樹中結(jié)點(diǎn)個數(shù)19
boolIsEmpty(); //判二叉樹空否?
BinTreeNode<T>*Parent(BinTreeNode<T>*t);
//求結(jié)點(diǎn)t的雙親
BinTreeNode<T>*LeftChild
(BinTreeNode<T>*t);
//求結(jié)點(diǎn)t的左子女
BinTreeNode<T>*RightChild(BinTreeNode<T>*t);
//求結(jié)點(diǎn)t的右子女
boolInsert
(Titem); //在樹中插入新元素
boolRemove(Titem); //在樹中刪除元素
boolFind(T&item); //判斷item是否在樹中
boolgetData(T&item); //取得結(jié)點(diǎn)數(shù)據(jù)20
BinTreeNode<T>*getRoot(); //取根
voidpreOrder(void(*visit)(BinTreeNode<T>*t)); //前序遍歷,visit是訪問函數(shù)
voidinOrder(void(*visit)(BinTreeNode<T>*t));
//中序遍歷,visit是訪問函數(shù)
voidpostOrder(void(*visit)(BinTreeNode<T>*t));
//后序遍歷,(*visit)是訪問函數(shù)
voidlevelOrder(void(*visit)(BinTreeNode<T>*t));
//層次序遍歷,visit是訪問函數(shù)};21完全二叉樹一般二叉樹的順序表示的順序表示二叉樹的順序表示1123456789
10
1412346789
12
14248910567312376489125101113221371531極端情形:只有右單支的二叉樹137153123二叉樹結(jié)點(diǎn)定義:每個結(jié)點(diǎn)有3個數(shù)據(jù)成員,data域存儲結(jié)點(diǎn)數(shù)據(jù),leftChild和rightChild分別存放指向左子女和右子女的指針。leftChilddatarightChilddataleftChildrightChild二叉鏈表二叉樹的鏈表表示24leftChilddataparentrightChildparentdataleftChildrightChild三叉鏈表二叉樹的鏈表表示每個結(jié)點(diǎn)增加一個指向雙親的指針parent,使得查找雙親也很方便。25二叉樹鏈表表示的示例AAABBBCCCDDDFFFEEErootrootroot26二叉樹遍歷二叉樹的遍歷就是按某種次序訪問樹中的結(jié)點(diǎn),要求每個結(jié)點(diǎn)訪問一次且僅訪問一次。設(shè)訪問根結(jié)點(diǎn)記作
V
遍歷根的左子樹記作
L
遍歷根的右子樹記作
R則可能的遍歷次序有
前序VLR
鏡像VRL
中序
LVR
鏡像RVL
后序LRV
鏡像RLV27中序遍歷二叉樹算法的框架是:若二叉樹為空,則空操作;否則中序遍歷左子樹(L);訪問根結(jié)點(diǎn)(V);中序遍歷右子樹(R)。遍歷結(jié)果
a+b*c
-
d
-
e/f中序遍歷(InorderTraversal)--/+*abcdef28二叉樹遞歸的中序遍歷算法template<classT>voidBinaryTree<T>::InOrder(BinTreeNode<T>*subTree,void(*visit)(BinTreeNode<T>*t)){if(subTree!=NULL){
InOrder(subTree->leftChild,visit);
//遍歷左子樹
visit(subTree); //訪問根結(jié)點(diǎn)
InOrder(subTree->rightChild,visit);
//遍歷右子樹
}};29前序遍歷二叉樹算法的框架是:若二叉樹為空,則空操作;否則訪問根結(jié)點(diǎn)(V);前序遍歷左子樹(L);前序遍歷右子樹(R)。遍歷結(jié)果
-+a*b
-
cd/ef前序遍歷(PreorderTraversal)--/+*abcdef30二叉樹遞歸的前序遍歷算法template<classT>voidBinaryTree<T>::PreOrder(BinTreeNode<T>*subTree,void(*visit)
(BinTreeNode<T>*t)){if(subTree!=NULL){
visit(subTree); //訪問根結(jié)點(diǎn)
PreOrder(subTree->leftChild,visit);
//遍歷左子樹
PreOrder(subTree->rightChild,visit);
//遍歷右子樹
}};31后序遍歷二叉樹算法的框架是:若二叉樹為空,則空操作;否則后序遍歷左子樹(L);后序遍歷右子樹(R);訪問根結(jié)點(diǎn)(V)。遍歷結(jié)果
abcd
-*+ef/-后序遍歷(PostorderTraversal)--/+*abcdef32二叉樹遞歸的后序遍歷算法template<classT>voidBinaryTree<T>::PostOrder(BinTreeNode<T>*subTree,void(*visit)(BinTreeNode<T>*t){if(subTree!=NULL){
PostOrder(subTree->leftChild,visit);
//遍歷左子樹
PostOrder(subTree->rightChild,visit); //遍歷右子樹
visit(subTree); //訪問根結(jié)點(diǎn)
}};33template<classT>intBinaryTree<T>::Size(BinTreeNode<T>*
subTree)const{//私有函數(shù):利用二叉樹后序遍歷算法計算二叉//樹的結(jié)點(diǎn)個數(shù)
if(subTree==NULL)return0; //空樹
elsereturn1+Size(subTree->leftChild)
+
Size(subTree->rightChild);};應(yīng)用二叉樹遍歷的事例34template<classT>intBinaryTree<T>::Height(BinTreeNode<T>*subTree)const{//私有函數(shù):利用二叉樹后序遍歷算法計算二叉//樹的高度或深度
if(subTree==NULL)return0; //空樹高度為0 else{ inti=Height(subTree->leftChild);intj=Height(subTree->rightChild); return(i<j)?j+1:i+1; };35利用二叉樹前序遍歷建立二叉樹以遞歸方式建立二叉樹。輸入結(jié)點(diǎn)值的順序必須對應(yīng)二叉樹結(jié)點(diǎn)前序遍歷的順序。并約定以輸入序列中不可能出現(xiàn)的值作為空結(jié)點(diǎn)的值以結(jié)束遞歸,此值在RefValue中。例如用“@”或用“-1”表示字符序列或正整數(shù)序列空結(jié)點(diǎn)。36層次序遍歷二叉樹就是從根結(jié)點(diǎn)開始,按層次逐層遍歷,如圖:遍歷順序abcdef--+/*層次序遍歷二叉樹的算法37這種遍歷需要使用一個先進(jìn)先出的隊列,在處理上一層時,將其下一層的結(jié)點(diǎn)直接進(jìn)到隊列(的隊尾)。在上一層結(jié)點(diǎn)遍歷完后,下一層結(jié)點(diǎn)正好處于隊列的隊頭,可以繼續(xù)訪問它們。算法是非遞歸的。38abcdeQa訪問a,進(jìn)隊Qa出隊訪問b,進(jìn)隊訪問c,進(jìn)隊bcQb出隊訪問d,進(jìn)隊cdQc出隊訪問e,進(jìn)隊deQd出隊eQe出隊39層次序遍歷的(非遞歸)算法template<classT>voidBinaryTree<T>::levelOrder(void(*visit)(BinTreeNode<T>*t)){if(root==NULL)return;
Queue<BinTreeNode<T>*>Q;
BinTreeNode<T>*p=root;
visit(p);Q.EnQueue(p);
while(!Q.IsEmpty()){Q.DeQueue(p);if(p->leftChild!=NULL){
40visit(p->leftChild);
Q.EnQueue(p->leftChild);}if(p->rightChild!=NULL){visit(p->rightChild);
Q.EnQueue(p->rightChild);}}};41由二叉樹的前序序列和中序序列可唯一地確定一棵二叉樹。例,前序序列{ABHFDECKG}
和中序序列{HBDFAEKCG
},構(gòu)造二叉樹過程如下:HBDFEKCGAEKCGABHDF42前序序列{ABHFDECKG}KCGEKCGABHDFEKCGABHFDEABHFDEABHFDCKG43如果前序序列固定不變,給出不同的中序序列,可得到不同的二叉樹。固定前序排列,選擇所有可能的中序排列,可能構(gòu)造多少種不同的二叉樹?61234578961237584944例如,有3個數(shù)據(jù){1,2,3
},可得5種不同的二叉樹。它們的前序排列均為
123,中序序列可能是321,231,213,132,123。前序序列為123,中序序列為
312
的二叉樹不存在。12312312312312345線索化二叉樹
(ThreadedBinaryTree)又稱為穿線樹。通過二叉樹的遍歷,可將二叉樹中所有結(jié)點(diǎn)的數(shù)據(jù)排列在一個線性序列中,可以找到某數(shù)據(jù)在這種排列下它的前驅(qū)和后繼。希望不必每次都通過遍歷找出這樣的線性序列。只要事先做預(yù)處理,將某種遍歷順序下的前驅(qū)、后繼關(guān)系記在樹的存儲結(jié)構(gòu)中,以后就可以高效地找出某結(jié)點(diǎn)的前驅(qū)、后繼。46線索(Thread)增加前驅(qū)Pred指針和后繼Succ指針的二叉樹predleftChilddatarightChildsuccabcdepredpredpredsuccsuccsuccD∧∧AE∧∧B∧∧C∧∧rootpredpredpredpredsuccsuccsuccsucc47這種設(shè)計的缺點(diǎn)是每個結(jié)點(diǎn)增加兩個指針,當(dāng)結(jié)點(diǎn)數(shù)很大時存儲消耗較大。改造樹結(jié)點(diǎn),將pred指針和succ指針壓縮到leftChild和rightChild的空閑指針中,并增設(shè)兩個標(biāo)志ltag和rtag,指明指針是指示子女還是前驅(qū)/后繼。后者稱為線索。ltag(或rtag)=0,表示相應(yīng)指針指示左子女(或右子女結(jié)點(diǎn));當(dāng)ltag(或rtag)=1,表示相應(yīng)指針為前驅(qū)(或后繼)線索。leftChildltagdatartagrightChild
48leftChildltagdatartagrightChild
abcdepredpredpredsuccsuccsuccDAEB∧C∧rootpredpredsuccsucc0000111111線索化二叉樹及其鏈表表示49線索化二叉樹的類定義template<classT>structThreadNode{ //線索二叉樹的結(jié)點(diǎn)類
intltag,rtag; //線索標(biāo)志
ThreadNode<T>*leftChild,*rightChild;
//線索或子女指針
Tdata; //結(jié)點(diǎn)數(shù)據(jù)
ThreadNode(constTitem)//構(gòu)造函數(shù)
:data(item),leftChild(NULL),rightChild(NULL),ltag(0),rtag(0){} };50template<classT>classThreadTree{ //線索化二叉樹類protected:
ThreadNode<T>*root; //樹的根指針
voidcreateInThread(ThreadNode<T>*current,
ThreadNode<T>*&pre);
//中序遍歷建立線索二叉樹
ThreadNode<T>*parent(ThreadNode<T>*t);
//尋找結(jié)點(diǎn)t的雙親結(jié)點(diǎn)public:
ThreadTree():root(NULL){} //構(gòu)造函數(shù)51voidcreateInThread();//建立中序線索二叉樹
ThreadNode<T>*First(ThreadNode<T>*current);
//尋找中序下第一個結(jié)點(diǎn)
ThreadNode<T>*Last(ThreadNode<T>*current);
//尋找中序下最后一個結(jié)點(diǎn)
ThreadNode<T>*Next(ThreadNode<T>*current);
//尋找結(jié)點(diǎn)在中序下的后繼結(jié)點(diǎn)
ThreadNode<T>*Prior(ThreadNode<T>*current);
//尋找結(jié)點(diǎn)在中序下的前驅(qū)結(jié)點(diǎn)
……… };52通過中序遍歷建立中序線索化二叉樹0A00B00C00D00E0rootpre==NULLcurrent530A0
1B00C00D00E0rootpre==NULLcurrent540A0
1B00C0
1D00E0rootprecurrent550A0
1B00C0
1D1
0E0rootprecurrent560A0
1B00C0
1D1
1E0rootprecurrent570A0
1B00C0
1D1
1E1
rootprecurrent580A0
1B00C1
1D1
1E1
rootpre后處理59樹的存儲表示A(B(E,F),C,D(G))
結(jié)點(diǎn)的utype域沒有畫出ABCDEFGABEFCDG1、廣義表表示60ABCDEFGdataparentABCDEFG-100011301234562、雙親表示樹中結(jié)點(diǎn)的存放順序一般不做特殊要求,但為了操作實(shí)現(xiàn)的方便,有時也會規(guī)定結(jié)點(diǎn)的存放順序。例如,可以規(guī)定按樹的前序次序存放樹中的各個結(jié)點(diǎn),或規(guī)定按樹的層次次序安排所有結(jié)點(diǎn)。
613、子女鏈表表示無序樹情形鏈表中各結(jié)點(diǎn)順序任意,有序樹必須自左向右鏈接各個子女結(jié)點(diǎn)。ABCDEFG123∧45∧6∧∧∧∧∧ABCDEFG0123456624、子女指針表示一個合理的想法是在結(jié)點(diǎn)中存放指向每一個子女結(jié)點(diǎn)的指針。但由于各個結(jié)點(diǎn)的子女?dāng)?shù)不同,每個結(jié)點(diǎn)設(shè)置數(shù)目不等的指針,將很難管理。為此,設(shè)置等長的結(jié)點(diǎn),每個結(jié)點(diǎn)包含的指針個數(shù)相等,等于樹的度(degree)。這保證結(jié)點(diǎn)有足夠的指針指向它的所有子女結(jié)點(diǎn)。但可能產(chǎn)生很多空閑指針,造成存儲浪費(fèi)。63等數(shù)量的鏈域ABCDEFG
datachild1child2child3childdABCDEFG空鏈域2n+1個645、子女-兄弟表示也稱為樹的二叉樹表示。結(jié)點(diǎn)構(gòu)造為:firstChild指向該結(jié)點(diǎn)的第一個子女結(jié)點(diǎn)。無序樹時,可任意指定一個結(jié)點(diǎn)為第一個子女。nextSibling指向該結(jié)點(diǎn)的下一個兄弟。任一結(jié)點(diǎn)在存儲時總是有順序的。若想找某結(jié)點(diǎn)的所有子女,可先找firstChild,再反復(fù)用nextSibling沿鏈掃描。
datafirstChildnextSibling65樹的子女-
兄弟表示
datafirstChildnextSiblingABCDEFGABCDGFE66堆(Heap)template<classT,classE>classMinPQ{//最小優(yōu)先級隊列類的定義public:
VirtualboolInsert(E&d)=0;
VirtualboolRemove(E&d)=0;};
優(yōu)先級隊列每次出隊列的是優(yōu)先權(quán)最高的元素。用堆實(shí)現(xiàn)其存儲表示,能夠高效運(yùn)作。67最小堆完全二叉樹順序表示
Ki≤K2i+1&&Ki≤K2i+2最大堆完全二叉樹順序表示Ki≥K2i+1&&Ki≥K2i+2090987877878454565653131532323531717堆的定義68堆的元素下標(biāo)計算由于堆存儲在下標(biāo)從0開始計數(shù)的一維數(shù)組中,因此在堆中給定下標(biāo)為i
的結(jié)點(diǎn)時如果
i=0,結(jié)點(diǎn)i
是根結(jié)點(diǎn),無雙親;否則結(jié)點(diǎn)i
的父結(jié)點(diǎn)為結(jié)點(diǎn)(i-1)/2);如果2i+1>n-1,則結(jié)點(diǎn)i
無左子女;否則結(jié)點(diǎn)i
的左子女為結(jié)點(diǎn)2i+1;如果2i+2>n-1,則結(jié)點(diǎn)i無右子女;否則結(jié)點(diǎn)i的右子女為結(jié)點(diǎn)
2i+2。69自下向上逐步調(diào)整為最小堆currentPos=i=3currentPos=i=25353171778780923456587i0923456587i將一組用數(shù)組存放的任意數(shù)據(jù)調(diào)整成堆70currentPos=i=15353171778780923456587i0923456587i71currentPos=i=05353171778780923456587i0923456587i09725353171778780923456587i0923456587i1773樹的應(yīng)用例子---決策樹74樹的應(yīng)用例子---文件系統(tǒng)搜索樹7576二叉搜索樹(BinarySearchTree)定義
二叉搜索樹或者是一棵空樹,或者是具有下列性質(zhì)的二叉樹:每個結(jié)點(diǎn)都有一個作為搜索依據(jù)的關(guān)鍵碼(Key),所有結(jié)點(diǎn)的關(guān)鍵碼互不相同。左子樹(如果非空)上所有結(jié)點(diǎn)的關(guān)鍵碼都小于根結(jié)點(diǎn)的關(guān)鍵碼。右子樹(如果非空)上所有結(jié)點(diǎn)的關(guān)鍵碼都大于根結(jié)點(diǎn)的關(guān)鍵碼。左子樹和右子樹也是二叉搜索樹。77351545504025102030二叉搜索樹例結(jié)點(diǎn)左子樹上所有關(guān)鍵碼小于結(jié)點(diǎn)關(guān)鍵碼;右子樹上所有關(guān)鍵碼大于結(jié)點(diǎn)關(guān)鍵碼;注意:若從根結(jié)點(diǎn)到某個葉結(jié)點(diǎn)有一條路徑,路徑左邊的結(jié)點(diǎn)的關(guān)鍵碼不一定小于路徑上的結(jié)點(diǎn)的關(guān)鍵碼。78如果對一棵二叉搜索樹進(jìn)行中序遍歷,可以按從小到大的順序,將各結(jié)點(diǎn)關(guān)鍵碼排列起來,所以也稱二叉搜索樹為二叉排序樹。二叉搜索樹的類定義#include<iostream.h>#include<stdlib.h>template<classE,classK>structBSTNode{ //二叉樹結(jié)點(diǎn)類
Edata; //數(shù)據(jù)域
BSTNode<E,K>*left,*right;//左子女和右子女79BSTNode(){left=NULL;right=NULL;
}
//構(gòu)造函數(shù)
BSTNode(constEd,BSTNode<E,K>*L=NULL,
BSTNode<E,K>*R=NULL){data=d;left=L;right=R;}
//構(gòu)造函數(shù)~BSTNode(){} //析構(gòu)函數(shù)
voidsetData(Ed){data=d;} //修改
EGetData(){returndata;} //提取
booloperator<(constE&x)
//重載:判小于
{returndata.key<x.key;}80booloperator>(constE&x)
//重載:判大于
{returndata.key>x.key;}booloperator==(constE&x)
//重載:判等于
{returndata.key==x.key;}};template<classE,classK>classBST{ //二叉搜索樹類定義public:BST(){root=NULL;
} //構(gòu)造函數(shù)
BST(Kvalue); //構(gòu)造函數(shù)
~BST(){}; //析構(gòu)函數(shù)
81boolSearch(constKx)const //搜索
{returnSearch(x,root)!=NULL;}
BST<E,K>&operator=(constBST<E,K>&R); //重載:賦值
voidMakeEmpty()
//置空
{MakeEmpty(root);root=NULL;}voidPrintTree()const{PrintTree(root);}//輸出
EMin(){returnMin(root)->data;} //求最小
EMax(){returnMax(root)->data;} //求最大
boolInsert(constE&e1)
//插入新元素
{returnInsert(e1,root);}82boolRemove(constKx){returnRemove(x,root);} //刪除含x的結(jié)點(diǎn)private:
BSTNode<E,K>*root; //根指針
KRefValue; //輸入停止標(biāo)志
BSTNode<E,K>* //遞歸:搜索
Search(constKx,BSTNode<E,K>*ptr);voidmakeEmpty(BSTNode<E,K>*&ptr); //遞歸:置空
voidPrintTree(BSTNode<E,K>*ptr)const; //遞歸:打印
BSTNode<E,K>* //遞歸:復(fù)制
Copy(constBSTNode<E,K>*ptr); 83BSTNode<E,K>*Min(BSTNode<E,K>*ptr);
//遞歸:求最小
BSTNode<E,K>*Max(BSTNode<E,K>*ptr);
//遞歸:求最大
boolInsert(constE&e1,BSTNode<E,K>*&ptr);
//遞歸:插入
boolRemove(constKx,BSTNode<E,K>*&ptr);
//遞歸:刪除};二叉搜索樹的類定義用二叉鏈表作為它的存儲表示,許多操作的實(shí)現(xiàn)與二叉樹類似。84二叉搜索樹的搜索算法在二叉搜索樹上進(jìn)行搜索,是一個從根結(jié)點(diǎn)開始,沿某一個分支逐層向下進(jìn)行比較判等的過程。它可以是一個遞歸的過程。假設(shè)想要在二叉搜索樹中搜索關(guān)鍵碼為x
的元素,搜索過程從根結(jié)點(diǎn)開始。如果根指針為NULL,則搜索不成功;否則用給定值
x
與根結(jié)點(diǎn)的關(guān)鍵碼進(jìn)行比較:若給定值等于根結(jié)點(diǎn)關(guān)鍵碼,則搜索成功,返回搜索成功信息并報告搜索到結(jié)點(diǎn)地址。85若給定值小于根結(jié)點(diǎn)的關(guān)鍵碼,則繼續(xù)遞歸搜索根結(jié)點(diǎn)的左子樹;否則。遞歸搜索根結(jié)點(diǎn)的右子樹。搜索45搜索成功搜索28搜索失敗35154550402510203086搜索過程是從根結(jié)點(diǎn)開始,沿某條路徑自上而下逐層比較判等的過程。搜索成功,搜索指針將停留在樹上某個結(jié)點(diǎn);搜索不成功,搜索指針將走到樹上某個結(jié)點(diǎn)的空子樹。設(shè)樹的高度為h,最多比較次數(shù)不超過h。87二叉搜索樹的插入算法為了向二叉搜索樹中插入一個新元素,必須先檢查這個元素是否在樹中已經(jīng)存在。在插入之前,先使用搜索算法在樹中檢查要插入元素有還是沒有。如果搜索成功,說明樹中已經(jīng)有這個元素,不再插入;如果搜索不成功,說明樹中原來沒有關(guān)鍵碼等于給定值的結(jié)點(diǎn),把新元素加到搜索操作停止的地方。8835154550402510203028插入新結(jié)點(diǎn)28二叉搜索樹的插入每次結(jié)點(diǎn)的插入,都要從根結(jié)點(diǎn)出發(fā)搜索插入位置,然后把新結(jié)點(diǎn)作為葉結(jié)點(diǎn)插入。89輸入數(shù)據(jù)
{53,78,65,17,87,09,81,15}535378537865537865175378658717537865091
溫馨提示
- 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)方式做保護(hù)處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負(fù)責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 2024年食品加工設(shè)備租賃合同
- 2024年精裝修住宅租賃節(jié)能減排合同
- 二零二五年度地下空間開發(fā)工程勘察設(shè)計合同3篇
- 2024年特許經(jīng)營合同:連鎖加盟
- 二零二五年度資產(chǎn)保全第三方擔(dān)保借款資產(chǎn)保全合同模板3篇
- 二零二五年度歷史文化保護(hù)項目合同文物保護(hù)與修復(fù)工程協(xié)議3篇
- 古詩詞誦讀《燕歌行 并序》說課稿 2024-2025學(xué)年統(tǒng)編版高中語文選擇性必修中冊001
- 二零二五年度辦公室裝修工程環(huán)保材料認(rèn)證合同6篇
- 二零二五年度工廠廢棄物綜合利用合同3篇
- 2024年華師大新版選擇性必修2語文下冊月考試卷
- 人教版培智一年級上生活適應(yīng)教案
- 推動架機(jī)械加工工序卡片
- RoHS檢測報告完整版
- 中國近現(xiàn)代史綱要(上海建橋?qū)W院)智慧樹知到答案章節(jié)測試2023年
- 同濟(jì)大學(xué)土力學(xué)試卷2023
- 南理工2023運(yùn)籌學(xué)試卷A及答案
- 【讀寫策略】如何編寫班史
- 重慶市綦江區(qū)篆塘鎮(zhèn)白坪村建筑用砂巖礦采礦權(quán)評估報告
- 行政查房情況記錄表
- JJG 1038-2008科里奧利質(zhì)量流量計
- 機(jī)械設(shè)計-課程設(shè)計鏈?zhǔn)捷斔蜋C(jī)傳動裝置
評論
0/150
提交評論