欧美精品一二区,性欧美一级,国产免费一区成人漫画,草久久久久,欧美性猛交ⅹxxx乱大交免费,欧美精品另类,香蕉视频免费播放

數(shù)據(jù)結構(C語言版) 第6章 樹

上傳人:da****ge 文檔編號:59522048 上傳時間:2022-03-03 格式:DOC 頁數(shù):63 大小:8.35MB
收藏 版權申訴 舉報 下載
數(shù)據(jù)結構(C語言版) 第6章 樹_第1頁
第1頁 / 共63頁
數(shù)據(jù)結構(C語言版) 第6章 樹_第2頁
第2頁 / 共63頁
數(shù)據(jù)結構(C語言版) 第6章 樹_第3頁
第3頁 / 共63頁

下載文檔到電腦,查找使用更方便

16 積分

下載資源

還剩頁未讀,繼續(xù)閱讀

資源描述:

《數(shù)據(jù)結構(C語言版) 第6章 樹》由會員分享,可在線閱讀,更多相關《數(shù)據(jù)結構(C語言版) 第6章 樹(63頁珍藏版)》請在裝配圖網上搜索。

1、第6章 樹 1樹的邏輯結構 核心關系:層次關系。 1.1 通俗定義 樹形表示法 廣義表表示法 A(B(E,F,G),C(H),D(I,J)) 樹是n(n>0)個結點的有限集,在任意一棵樹中: 1 有且只有一個特定的根(root)結點; 2 當n>1時,其余結點可分為m(m>0)個互不相交的有限子集T1,T2...Tm,每個子集是一棵樹,稱為子樹。 1.2 形式定義 D={ai | ai∈ElemSet,i=1,…,n} 二元關系S的定義: 當n=1時,S=φ; 當n>1時: 1 根的特殊地位 唯一無前驅的元素:根結點(root)

2、; 2 結點的劃分 可將D-{root}劃分成m(m>0)個互不相交的子集D1,D2,…Dm,對任意子集Di,唯一存在xi∈Di,有∈S; 3 關系的劃分 對應D-{root}的劃分,可將 S-{,…,}唯一劃分成互不相交的子集S1,S2,...,Sm,對于任意i, Si是Di上的二元關系。(Di,Si)是一棵樹,稱為根root的子樹。 示例: D={A,B,C,D,E,F,G,H,I,J,K} S={,,, ,,, , <

3、D,I>, } root: A D-{A}可分為: D1={B,E,F,G} D2={C,H} D3={D,I,J} S-{,,}可分為: S1: {,,} S2: {} S3: {,} 1.3 概念 1.3.1 若干角色 雙親、孩子 一個序偶中的直接前驅、直接后繼。 祖先、子孫 一個序偶集合中的前驅、后繼。 兄弟 具有相同直接前驅的數(shù)據(jù)元素。 堂兄弟 具有相同前驅的數(shù)據(jù)元素。 1.3.2 與“度”相關的概念 結點的度 結點擁有的子樹數(shù)。

4、 葉子結點 度為0的結點。 分支結點 度不為0的結點。 樹的度 樹內所有結點的度的最大值。 1.3.3 與“深度”相關的概念 結點深度 根結點的深度為1 若某結點深度為i, 則其子結點深度為i+1 樹的深度 樹中結點的最大深度。 或 空樹深度為0; 非空樹深度等于子樹深度的最大值加1。 1.3.4 其他概念 有序樹、無序樹 左右結點是否等價 森林 m棵互不相交的樹的集合。 m=0:空集; m=1:樹 1.4 基本操作 構造、析構、遍歷、查找、插入、刪除、 2 二叉樹的邏輯結構 2.1 定義 結構簡化、

5、概念強化:左子樹、右子樹。 樹形表示法 廣義表表示法 A(B(D),C(F(,E),G)) 2.2 二叉樹的形態(tài) 具有n個結點的不同形態(tài)的二叉樹有多少顆? 二叉樹相似:二者都為空;或它們的左右子樹相似。 例:n個結點的相似樹的個數(shù) template int BinTree::GetTreeCount(int n) { int left,count=0; if(n==0 || n==1) return(1); for(left=n-1; left>=0; left--) count += GetT

6、reeCount(left) * GetTreeCount(n-1-left); return(count); } 2.3 性質 2.3.1 性質1 若某二叉樹的葉子結點數(shù)為n0,度為2的結點數(shù)為n2,則n0=n2+1。 試題例:若一棵m叉樹中度為i的結點有Ni個,則該樹的葉結點有 (N1+2N2+…+mNm+1)-(N1+N2+…+Nm) 個。 2.3.2 性質2 二叉樹的第i層上的結點數(shù)最多為2i-1。 2.3.3 性質3 深度為k的二叉樹中結點總數(shù)最多為2k-1。 滿二叉樹 完全二叉樹 深度為k,共有2k-1個結點。 深

7、度為k,前k-1層是滿二叉樹,第k層的結點從左至右依次排列。 2.3.4 性質4 具有n個結點的完全二叉樹的深度為int(log2n)+1。 2.3.5 性質5 有n個結點的完全二叉樹,結點從0順序標號,則: 1、若i>0,i的雙親結點是(i-1)/2。 2、若2i+1

8、結構存于數(shù)組A[0…100]中,葉子結點的最小編號是 。 A、32 B、33 C、34 D、35 試題例:具有101個結點的完全二叉樹,度為1的結點有 個。 3.1.2 性能分析 滿/完全二叉樹:存儲效率最高,插入、刪除方便。 非完全二叉樹的處理方法: 非完全二叉樹 修補結構 不存在的結點,用特殊符號標識。 退化的二叉樹及其存儲效果: 評價:空間浪費大,不靈活。 3.2 鏈式結構 3.2.1 二叉鏈表結構 Template struct BinTreeNod

9、e { T data; BinTreeNode *lchild, *rchild; }; template class BinTree { BinTreeNode* m_Root; …… } 邏輯結構 存儲結構 空指針域有多少? 3.2.2 三叉鏈表結構 Template struct BinTreeNode { T data; BinTreeNode *parent; BinTreeNode *lchild; BinTreeNode *rchild; };

10、 邏輯結構 存儲結構 3.2.3 三叉靜鏈表結構 Template struct BinTreeNode { T data; int parent,lchild, rchild; }; template class BinTree { int m_Root; // 根結點的下標 BinTreeNode* m_Data; // 結點的存儲空間 …… } 邏輯結構 存儲結構 第6章 樹 專題:二叉樹存儲結構的遐想 一、結構定義 enum T

11、ag {LEFT,NOLEFT}; Template struct BinTreeNode { Tag type; T data; BinTreeNode *child, *right; }; ①若結點*p有左孩子,則p->type=LEFT;否則p->type=NOLEFT; ②若結點*p是根結點,則p->right=NULL; ③若結點*p是葉子結點,則p->child=NULL; ④若結點*p有且僅有一個子結點*q,則 p->child=q; q->right=p; ⑤若結點*p有左孩子*q,有右孩子*r,則

12、 p->child=q; q->right=r; r->right=p; 二、結構示例 二叉鏈表結構 三、基本操作 ①求結點*p的左孩子*p_LeftChild。 if(p->type==LEFT) p_LeftChild=p->child; ②求結點*p的右孩子*p_RightChild。 if(p->type==LEFT && p->child->right!=p) p_RightChild=p->child->right; 或 if(p->type==NOLEFT && p->child!=NULL)

13、 p_RightChild=p->child; ③當p->right!=NULL時,求結點*p的父結點*p_Parent。 設q=p->right if(q->child==p) p_Parent=q; // *p是獨左/右子 if(q->child==NULL) p_Parent=q->right; // *p和*q是兄弟 if(q->child!=NULL && q->child->right==p) p_Parent=q; //q->child指向左孩子,p指向右孩子 if(q->

14、right==NULL) p_Parent=q; // *p_Parent是根 if(q->right!=NULL && q->right->child==p) p_Parent=q->right; // *p是左孩子,*q是右孩子 四、結構優(yōu)點 查找父結點的時間復雜度為O(1); 遍歷二叉樹,不再需要棧,提高了指針的利用率。 第6章 樹 4 遍歷二叉樹 遍歷:每個結點訪問且只訪問一次。 template class BinTree { BinTreeNode* m_Root;

15、 public: void TraverseDFS(int kind); // 深度遍歷二叉樹 void TraverseBFS(); // 層次(廣度)遍歷二叉樹 private: void TraversePre(BinTreeNode *p); void TraverseMid(BinTreeNode *p); void TraversePost(BinTreeNode *p); } 4.1 深度遍歷(遞歸):先序、中序、后序 4.1.1 算法 算法思路:二叉樹 = 根結點 + 左子樹 + 右子樹 將二

16、叉樹的遍歷轉變?yōu)樽訕涞谋闅v。 先序算法:根左右 中序算法:左根右 后序算法:左右根 ①若二叉樹為空,結束 ②訪問根結點 ③先序遍歷左子樹 ④先序遍歷右子樹 ①若二叉樹為空,結束 ②中序遍歷左子樹 ③訪問根結點 ④中序遍歷右子樹 ①若二叉樹為空,結束 ②后序遍歷左子樹 ③后序遍歷右子樹 ④訪問根結點 先序序列:根左右 ABDECF 中序序列:左根右 DBEAFC 后序序列:左右根 DEBFCA 遍歷序列與二叉樹不是一一對應的。 例:若前序序列為123,對應的二叉樹有5種。 試題

17、例:試找出分別滿足下列條件的所有二叉樹 ①先序序列和中序序列相同 ②中序序列和后序序列相同 ③先序序列和后序序列相同 4.1.2 算法實現(xiàn)、調試、理解 template void BinTree::TraverseDFS(int kind) // 深度遍歷二叉樹 { switch(kind) { case 1: TraversePre (m_Root); break; case 2: TraverseMid (m_Root); break; case 3: TraversePost(m_Root); break; }

18、 cout< void BinTree::TraversePre(BinTreeNode *p) { ①if(p==NULL) return; ②cout<data; ③TraversePre(p->lchild); ④TraversePre(p->rchild); } 評價: 邏輯極其清晰。 但只有邏輯能力強的人,才能理解。 調試、理解程序: 調試技術: 觀察調用棧alt+7 P(&A) ①② =>'A'

19、③P(&B) ①② =>'B' ③P(&D) ①② =>'D' ③P(NULL);④P(NULL) ④P(&E) ①② =>'E' ③P(NULL);④P(NULL) ④P(&C) ①② =>'C' ③P(&F) ①② =>'F' ③P(NULL);④P(NULL) ④P(NULL) 性能分析:(設結點數(shù)為n,樹深度為k) 時間復雜度 O(n) 空間復雜度 O(k) 棧的

20、最大深度等于樹深度 4.1.3 算法的另一種實現(xiàn) template vector BinTree::TraverseDFS2(int kind) { vector L; switch(kind) { case 1: TraversePre (m_Root,L); break; case 2: TraverseMid (m_Root,L); break; case 3: TraversePost(m_Root,L); break; } return L; } template

21、 void BinTree::TraverseMid (BinTreeNode *p,vector &L) { if(p==NULL) return; TraverseMid(p->lchild,L); L.push_back(p->data); TraverseMid(p->rchild,L); } 4.2 層次(廣度)遍歷二叉樹 4.2.1 算法 ①將根指針加入隊列; ②取隊首元素,設為p;訪問結點*p; ③若*p存在左孩子,將其加入隊列; 若*p存在右孩子,將其加入隊列; ④若隊列不空,則循環(huán)②③,否則遍歷完成。

22、 template void BinTree::TraverseBFS() // 層次遍歷二叉樹 { BinTreeNode *p; queue *> Q; // Q為指針隊列 if(m_Root==NULL) return; Q.push(m_Root); // 將m_Root進隊列 while(!Q.empty()) { p=Q.front(); // 將隊首元素賦給p Q.pop(); // 隊首元素出隊列 cout<data;

23、if(p->lchild) Q.push(p->lchild); if(p->rchild) Q.push(p->rchild); } cout<

24、avFlag{START,LEFT,RIGHT}; template class Status // 遍歷狀態(tài)類 { public: BinTreeNode *p; // 遍歷中的位置 TravFlag flag; // 遍歷中的方向 void NextFlag() { if(flag==START) { flag=LEFT; return; } if(flag==LEFT ) { flag=RIGHT; return; } } }; 4.3.2 算法及實現(xiàn) ①將起點狀態(tài){m_Ro

25、ot,START}進棧; ②取棧頂狀態(tài),設為s; ③若s的當前方向是START, 若存在左孩子,則將{左孩子,START}進棧; 若s的當前方向是START, 若存在右孩子,則將{右孩子,START}進棧; ④若棧不空,則循環(huán)②③,否則遍歷完成。 template void BinTree::TraverseDFS(int kind) { stack > S; // 遍歷狀態(tài)棧 Status s1={m_Root,START}; S.push(s1); while( !S.empty(

26、) ) { Status &s=S.top(); // s是棧頂狀態(tài)的引用 Status nexts; switch (s.flag) { case START: if(kind==1) cout<data<<" "; if(s.p->lchild) // 若存在左孩子,向左孩子方向前進 { nexts.p=s.p->lchild; nexts.flag=START; S.push(nexts); } s.NextFla

27、g(); // 調整棧頂狀態(tài)的方向 break; case LEFT: if(kind==2) cout<data<<" "; if(s.p->rchild) // 若存在左孩子,向左孩子方向前進 { nexts.p=s.p->rchild; nexts.flag=START; S.push(nexts); } s.NextFlag(); // 調整棧頂狀態(tài)的方向 break; case R

28、IGHT: if(kind==3) cout<data<<" "; S.pop(); } } cout< int BinTree

29、>::GetCount() { return Count(m_Root); } template int BinTree::Count(BinTreeNode *p) { int left,right; if(p==NULL) return(0); left= Count(p->lchild); right=Count(p->rchild); return 1+left+right; } 思考:下列二叉樹計數(shù)算法是否有錯? template int BinTree::Count(Bin

30、TreeNode *p) { int s=0; if(p) { s++; count(p->lchild); count(p->rchild); return(s); } } // 統(tǒng)計葉子結點的個數(shù) template int BinTree::GetLeafCount() { return LeafCount(m_Root); } template int BinTree::LeafCount(BinTreeNode *p) { int lef

31、t,right; if(p==NULL) return(0); if(root->lchild==NULL && root->rchild==NULL) return 1; left= Count(p->lchild); right=Count(p->rchild); return left+right; } 5.1.2 層次遍歷模式 template int BinTree::GetCount() { BinTreeNode *p; int count=0; queue

32、 *> Q; // Q為指針隊列 if(m_Root==NULL) return; Q.push(m_Root); // 將m_Root進隊列 while(!Q.empty()) { p=Q.front(); // 將隊首元素賦給p Q.pop(); // 隊首元素出隊列 count++; if(p->lchild) Q.push(p->lchild); if(p->rchild) Q.push(p->rchild); } return count; } 5.2 析構函數(shù):釋放

33、整個樹的空間 5.2.1 深度遍歷模式 template void BinTree::Free() { DoFree(m_Root); m_Root=NULL; } template void BinTree::DoFree(BinTreeNode *p) { if(p==NULL) return; DoFree( p->lchild ); // 釋放左子樹 DoFree( p->rchild ); // 釋放右子樹 delete p; } 5.2.2 層次遍歷模式

34、 template void BinTree::Free() { BinTreeNode *p; queue *> Q; // Q為指針隊列 if(m_Root==NULL) return; Q.push(m_Root); // 將m_Root進隊列 while(!Q.empty()) { p=Q.front(); // 將隊首元素賦給p Q.pop(); // 隊首元素出隊列 if(p->lchild) Q.push(p->lchild);

35、 if(p->rchild) Q.push(p->rchild); delete p; } } 5.3 計算二叉樹的深度/高度 template int BinTree::GetHeight() { return Height(m_Root); } template int BinTree::Height(BinTreeNode *p) { int left,right; if(p==NULL) return(0); left =Height(p->lchild); ri

36、ght=Height(p->rchild); if (left>right) return left+1; return right+1; } 思考:與廣義表的深度算法的相似之處。 5.4 交換樹中每個結點的左右子樹 template void BinTree::Exchange() { DoExchange(m_Root); } template void BinTree::DoExchange(BinTreeNode *p) { if(p==NULL) return; DoExchan

37、ge( p->lchild ); DoExchange( p->rchild ); BinTreeNode *tmp=p->lchild; p->lchild=p->rchild; p->rchild=tmp; } 5.5 查找函數(shù) 5.5.1 查找值為e的結點的地址 template BinTreeNode *BinTree::Search(T e) { return DoSearch(m_Root, e); } template BinTreeNode *Bi

38、nTree::DoSearch(BinTreeNode *p,T e) { if(p==NULL) return(NULL); if(p->data==e) return(p); BinTreeNode *q=DoSearch(p->lchild, e); if(q) return q; return DoSearch(p->rchild, e); } 5.5.2 查找值為e的結點的父結點的地址 template BinTreeNode *BinTree::SearchParent(T e)

39、 { return DoSearchParent(m_Root, e); } template BinTreeNode *BinTree::DoSearchParent(BinTreeNode *p,T e) { if(p==NULL) return NULL; if(p->lchild && p->lchild->data==e) return p; if(p->rchild && p->rchild->data==e) return p; BinTreeNode *q=DoSearchParent(p->lchild,

40、 e); if(q) return q; return DoSearchParent(p->rchild, e); } 5.5.3 思考題: 查找所有值為e的結點,刪除以該結點為根的子樹。 template void BinTree::SearchDelete(T e) { return DoSearchDelete(m_Root, e); } template void BinTree::DoSearchDelete(BinTreeNode *p,T e) { if(p==NULL) return

41、 NULL; if(p->lchild && p->lchild->data==e) { DoFree(p->lchild); p->lchild=NULL; } if(p->rchild && p->rchild->data==e) { DoFree(p->rchild); p->rchild=NULL; } DoSearchDelete(p->lchild, e); DoSearchDelete(p->rchild, e); } 5.6 查找極值結點 template BinTreeNode * BinTree

42、>::GetMax() { return GetMax(m_Root); } template BinTreeNode * BinTree::GetMax(BinTreeNode *p) { if(p==NULL) return NULL; BinTreeNode *plMax, *prMax, *pMax; plMax = GetMax(p->lchild); prMax = GetMax(p->rchild); if(plMax==NULL && prMax==NULL) return p; // 左右子樹均為空

43、 if(plMax==NULL) // 右子樹不為空 if(prMax->data > p->data) return prMax; else return p; if(prMax==NULL) // 右子樹不為空 if(plMax->data > p->data) return plMax; else return p; // 左右子樹均不為空 if(plMax->data > prMax->data) pMax=plMax; else pMax=prMax; if(p->data > pMax->data) return p; el

44、se return pMax; } 5.7 構造函數(shù)(參數(shù):帶空指針標記的先序序列) 先序序列: ABDECF 帶空指針標記的先序序列: ABD**E**CF*** template BinTree::BinTree(vector &pre) { m_pre=pre; m_prei=0; // 完成數(shù)據(jù)準備 m_Root=DoCreateByPre(); // 調用核心函數(shù) } template BinTreeNode *BinTree::DoCreateByPre()

45、 { T e=m_pre[m_prei]; m_prei++; if(e=='*') return(NULL); BinTreeNode *p=new BinTreeNode; p->data=e; p->lchild=DoCreateByPre(); // 建左子樹 p->rchild=DoCreateByPre(); // 建右子樹 return(p); } 5.8 構造函數(shù)(參數(shù):先序序列、中序序列) 5.8.1 先序序列、中序序列與二叉樹的對應關系 先序: A B H F D E C K G 中序: H B D

46、 F A E K C G 取A 取B 取H 取F 取D 取E 取C 取K 取G 5.8.2 算法與實現(xiàn) 遞歸算法的常識: 組成 ①分解策略,②最終小問題的解決方案。 外觀 大小問題必須具有完全相同的形式。 template BinTree::BinTree(vector &pre,vector &mid) { m_pre=pre; m_mid=mid; // 完成數(shù)據(jù)準備 int n=m_pre.size(); m_Root=DoCreateByPreMid(0,0,

47、 n); // 調用核心函數(shù) } template BinTreeNode *BinTree::DoCreateByPreMid(int ipre,int imid,int n) { if(n==0) return(NULL); // 最終解決方案 BinTreeNode *p=new BinTreeNode; p->data = m_pre[ipre]; for(int i=0; i

48、; p->lchild = DoCreateByPreMid(ipre+1, imid, i); p->rchild = DoCreateByPreMid(ipre+i+1,imid+i+1,n-i-1); return p; } 5.8.3 延伸思考 利用后序序列、中序序列,可以構造二叉樹嗎? 利用先序序列、后序序列,可以構造二叉樹嗎? 5.9 拷貝構造函數(shù) template BinTree::BinTree(BinTree &tree) { m_Root=DoCopy(tree.m_Root); }

49、template BinTreeNode * BinTree::DoCopy(BinTreeNode *p) { if(p==NULL) return NULL; BinTreeNode *newp=new BinTreeNode; newp->data=p->data; newp->lchild= DoCopy(p->lchild); // 復制左子樹 newp->rchild= DoCopy(p->rchild); // 復制右子樹 return newp; } 5.10 表達式樹 5

50、.10.1 存儲結構的優(yōu)化 字符串存儲結構:在運算中,需要識別運算符、運算數(shù);比較優(yōu)先級。效率低。 表達式樹的邏輯結構 運算符 運算對象 左操作數(shù) 右操作數(shù) 分支結點 葉子結點 左子樹 右子樹 例:a+b*c-d 5.10.2 表達式樹的類定義 class BinTree { static char m_Priority[4][4]; BinTreeNode* m_Root; public: BinTree(); BinTree(char mid[]); ~BinTree(); vo

51、id Free(); // 釋放整個樹的空間 void TraverseDFS(int kind); // 深度遍歷二叉樹 double Value(); …………… } 5.10.3 遍歷的意義:求值 先序:前綴表達式 中序:中綴表達式 后序:后綴表達式 // 一次建樹,多次高效計算、轉換方便 double BinTree::Value() { return DoValue(m_Root); } double BinTree::DoValue(BinTreeNode* p) { if(p->type==OPD) return(p->data);

52、 double opd1 = DoValue(p->lchild); double opd2 = DoValue(p->rchild); return Compute(opd1,p->op,opd2); } 5.10.4 建樹算法 設已有表達式樹的根指針為m_Root,則每建立一個運算符結點*op和運算數(shù)結點*opd3時,結點關系需做如下調整: 若*op優(yōu)先級高:*op作*m_Root的右孩子。 a+b a+b*c op->lchild=m_Root->rchild; op->rchild=opd3; m_Root->rchild=

53、op; 若*op優(yōu)先級低:*m_Root作為*op的左孩子。 a*b a*b-c op->lchild=m_Root; op->rchild=opd3; m_Root=op; 5.10.5 算法實現(xiàn) 理解次序: 1 設表達式不含括號,忽略代碼中所有紅字部分,即實現(xiàn)簡單表達式的建樹過程。 2 考慮任一運算數(shù)都可能是括號表達式的情形,則紅字部分的遞歸調用實現(xiàn)了復雜表達式的建樹過程。 BinTree::BinTree(char mid[]) { m_midi=0; // 與mid配合使用的下標變量 m_Root = DoCreate(

54、mid); } BinTreeNode* BinTree::DoCreate(char mid[]) { // 初始化最初始的樹(3個結點) BinTreeNode *root,*opd1,*opd2; if(mid[m_midi]=='(') { m_midi++; opd1=DoCreate(mid); } else { opd1=new BinTreeNode; // 第1個運算數(shù)結點 opd1->type=OPD; opd1->data=mid[m_midi++]-'0'; opd1->lchild=

55、opd1->rchild=NULL; } root = new BinTreeNode; // 第1個運算符結點 root->type=OP; root->op=mid[m_midi++]; if(mid[m_midi]=='(') { m_midi++; opd2=DoCreate(mid); } else { opd2=new BinTreeNode; // 第2個運算數(shù)結點 opd2->type=OPD; opd2->data=mid[m_midi++]-'0'; opd2->lchild=

56、opd2->rchild=NULL; } root->lchild=opd1; root->rchild=opd2; // 每躺循環(huán)增加2個結點: *op和*opd3 while(mid[m_midi]) { if(mid[m_midi]==')') { m_midi++; break; } BinTreeNode *op=new BinTreeNode; // 運算符結點 op->type=OP; op->op=mid[m_midi++]; BinTreeNode *opd3; if(mid[m_midi]==

57、'(') { m_midi++; opd3=DoCreate(mid); } else { opd3=new BinTreeNode; // 運算數(shù)結點 opd3->type=OPD; opd3->data=mid[m_midi++]-'0'; opd3->lchild=opd3->rchild=NULL; } // 比較*root和*op的優(yōu)先級 if(Precede(root->op, op->op)=='<') if(Precede(root->op, op->o

58、p)=='<') { op->lchild=root->rchild; op->rchild=opd3; root->rchild=op; } else { op->lchild=root; op->rchild=opd3; root=op; } } return root; } 5.9.6 體會 棧和樹是等價的。 作業(yè)與上機 1 二叉樹類 建立二叉樹類,應包含以下成員函數(shù):構造、深度/層次遍歷、查找、深度。 方向1:盡可能的豐富二叉樹類的成員函數(shù); 方向2:設計更多

59、拷貝構造函數(shù)(實現(xiàn)類型轉換功能),如:將二叉樹的順序結構轉換為二/三叉鏈表;將序偶的集合結構轉換為二/三叉鏈表; 方向3:建立表達式樹類; 方向4:其他二叉樹類的應用算法:計算寬度、結點坐標… 方向5:趣味賽題的程序結構的共性研究; 2 二叉線索樹類 建立二叉樹線索樹類,應包含以下成員函數(shù):構造、某種線索化、遍歷。 方向1:三類線索樹、樹類,構成類的繼承體系 方向2:線索樹的插入、刪除等算法 3 樹類 采用二叉鏈表結構,建立樹類,應包含以下成員函數(shù):構造、先/后根遍歷、結點/樹的度、深度。 4 哈夫曼類 采用哈夫曼樹結構,計算哈夫曼編碼

60、;實現(xiàn)壓縮與解壓縮功能。 第6章 樹 6 線索二叉樹 如何快捷地找出結點在遍歷過程中的前驅、后繼?如何保存遍歷過程得到的先后次序? 方法 評價 每個結點增加前驅域、后繼域。 浪費空間 利用結點的空鏈域存儲前驅指針和后繼指針。 問題:如何區(qū)別左孩子指針和前驅指針? 對策:增加標記域。 質疑:浪費空間? 解釋:標記數(shù)據(jù)有許多靈巧的存儲方法。 6.1 線索二叉樹的存儲結構 中序線索樹示例: 6.2 線索二叉樹類 // 二叉線索樹結點 template struct BinThrTreeNode { BinThrTree

61、NodeType ltype,rtype; T data; BinThrTreeNode *lchild,*rchild; }; // 二叉“中序”線索樹的類 template class BinThrTree { BinThrTreeNode *m_Root; public: BinThrTree(); ~BinThrTree(); BinThrTree(vector &pre); // 根據(jù)pre創(chuàng)建二叉樹 void Free(); // 釋放整個樹的空間 void MT_Ord

62、er(); // 中序線索化 // 求中序遍歷中的后繼、前驅 BinThrTreeNode * MT_GetNext(BinThrTreeNode *p); BinThrTreeNode * MT_GetPrev(BinThrTreeNode *p); void MT_Travese(); // 不使用棧的中序遍歷 // 求父結點地址 BinThrTreeNode *MT_GetParent(BinThrTreeNode *p); ………… } 6.3 線索化 6.3.1 利用遞歸遍歷程序模式,實現(xiàn)中序

63、線索化 設已有二叉樹所有結點的ltype、rtype為LINK ①遍歷指針p依次指向各結點,m_lastp緊隨其后,指向上一個被訪問的結點。 ②每次指針變調整前,進行*p的前驅線索化、*m_lastp的后繼線索化。 template void BinThrTree::MT_Order() // 中序線索化 { if(m_Root==NULL) return; m_lastp=NULL; MT_DoOrder(m_Root); m_lastp->rtype=THREAD; //最后遍歷的是最右下的葉子 } template

64、ss T> void BinThrTree::MT_DoOrder(BinThrTreeNode *p) { if(p==NULL) return; // 左子樹中序線索化 if(p->ltype==LINK) MT_DoOrder( p->lchild ); if(p->lchild==NULL) // *p的前驅線索化 { p->ltype=THREAD; p->lchild=m_lastp; } if(m_lastp && m_lastp->rchild==NULL) // m_lastp的后繼線索化 { m_lastp->rt

65、ype=THREAD; m_lastp->rchild=p; } m_lastp = p; // 右子樹中序線索化 if(p->rtype==LINK) MT_DoOrder( p->rchild ); } 時間/空間復雜度: 等同于原遍歷算法。 6.3.2 利用非遞歸遍歷程序模式 注意:在先序、中序、后續(xù)線索化中,最后一個結點的后繼線索化,需要具體分析。 先序、中序 m_lastp->rtype=THREAD; 后序 if(m_lastp->rchild) m_lastp->rtype=LINK; else

66、 m_lastp->rtype=THREAD; 6.4 中序線索樹中的若干應用 6.4.1 檢索結點 求*p的后繼 若p->RTag=THREAD:后繼為*p->rchild 若p->RTag=LINK :后繼為*p的右子樹中最左下結點 求*p的前驅 若p->LTag=THREAD:前驅為*p->lchild 若p->LTag=LINK :前驅為*p的左子樹中最右下結點 // 求*p的后繼 template BinThrTreeNode* BinThrTree::MT_GetNext(BinThrTreeNode *p) { if(p->rtype==THREAD) return p->rchild; for(p=p->rchild; p->ltype==LINK; p=p->lchild); return p; } // 求*p的前驅 template BinThrTreeNode* BinThrTree::MT_GetPrev(BinThrTreeNo

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

相關資源

更多
正為您匹配相似的精品文檔
關于我們 - 網站聲明 - 網站地圖 - 資源地圖 - 友情鏈接 - 網站客服 - 聯(lián)系我們

copyright@ 2023-2025  zhuangpeitu.com 裝配圖網版權所有   聯(lián)系電話:18123376007

備案號:ICP2024067431-1 川公網安備51140202000466號


本站為文檔C2C交易模式,即用戶上傳的文檔直接被用戶下載,本站只是中間服務平臺,本站所有文檔下載所得的收益歸上傳人(含作者)所有。裝配圖網僅提供信息存儲空間,僅對用戶上傳內容的表現(xiàn)方式做保護處理,對上載內容本身不做任何修改或編輯。若文檔所含內容侵犯了您的版權或隱私,請立即通知裝配圖網,我們立即給予刪除!