數(shù)據(jù)結構(C語言版) 第6章 樹
《數(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,有
3、D,I>,
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
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 9、e
{ T data;
BinTreeNode 10、
邏輯結構
存儲結構
3.2.3 三叉靜鏈表結構
Template 11、ag {LEFT,NOLEFT};
Template 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 15、
public:
void TraverseDFS(int kind); // 深度遍歷二叉樹
void TraverseBFS(); // 層次(廣度)遍歷二叉樹
private:
void TraversePre(BinTreeNode 16、叉樹的遍歷轉變?yōu)樽訕涞谋闅v。
先序算法:根左右
中序算法:左根右
后序算法:左右根
①若二叉樹為空,結束
②訪問根結點
③先序遍歷左子樹
④先序遍歷右子樹
①若二叉樹為空,結束
②中序遍歷左子樹
③訪問根結點
④中序遍歷右子樹
①若二叉樹為空,結束
②后序遍歷左子樹
③后序遍歷右子樹
④訪問根結點
先序序列:根左右
ABDECF
中序序列:左根右
DBEAFC
后序序列:左右根
DEBFCA
遍歷序列與二叉樹不是一一對應的。
例:若前序序列為123,對應的二叉樹有5種。
試題 17、例:試找出分別滿足下列條件的所有二叉樹
①先序序列和中序序列相同
②中序序列和后序序列相同
③先序序列和后序序列相同
4.1.2 算法實現(xiàn)、調試、理解
template 18、
cout< 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 21、
void BinTree 22、
template 23、if(p->lchild) Q.push(p->lchild);
if(p->rchild) Q.push(p->rchild);
}
cout< 24、avFlag{START,LEFT,RIGHT};
template 25、ot,START}進棧;
②取棧頂狀態(tài),設為s;
③若s的當前方向是START,
若存在左孩子,則將{左孩子,START}進棧;
若s的當前方向是START,
若存在右孩子,則將{右孩子,START}進棧;
④若棧不空,則循環(huán)②③,否則遍歷完成。
template 26、) )
{ Status 27、g(); // 調整棧頂狀態(tài)的方向
break;
case LEFT:
if(kind==2) cout< 28、IGHT:
if(kind==3) cout< 29、>::GetCount()
{ return Count(m_Root); }
template 30、TreeNode 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 32、 33、整個樹的空間
5.2.1 深度遍歷模式
template 34、
template 35、 if(p->rchild) Q.push(p->rchild);
delete p;
}
}
5.3 計算二叉樹的深度/高度
template 36、ght=Height(p->rchild);
if (left>right) return left+1;
return right+1;
}
思考:與廣義表的深度算法的相似之處。
5.4 交換樹中每個結點的左右子樹
template 37、ge( p->lchild );
DoExchange( p->rchild );
BinTreeNode 38、nTree 39、
{ return DoSearchParent(m_Root, e); }
template 40、 e);
if(q) return q;
return DoSearchParent(p->rchild, e);
}
5.5.3 思考題: 查找所有值為e的結點,刪除以該結點為根的子樹。
template 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 42、>::GetMax()
{ return GetMax(m_Root); }
template 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 45、
{ T e=m_pre[m_prei]; m_prei++;
if(e=='*') return(NULL);
BinTreeNode 46、 F A E K C G
取A
取B
取H
取F
取D
取E
取C
取K
取G
5.8.2 算法與實現(xiàn)
遞歸算法的常識:
組成
①分解策略,②最終小問題的解決方案。
外觀
大小問題必須具有完全相同的形式。
template 47、 n); // 調用核心函數(shù)
}
template 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 49、template 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 61、NodeType ltype,rtype;
T data;
BinThrTreeNode 62、er(); // 中序線索化
// 求中序遍歷中的后繼、前驅
BinThrTreeNode 63、線索化
設已有二叉樹所有結點的ltype、rtype為LINK
①遍歷指針p依次指向各結點,m_lastp緊隨其后,指向上一個被訪問的結點。
②每次指針變調整前,進行*p的前驅線索化、*m_lastp的后繼線索化。
template 64、ss T>
void BinThrTree 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
- 溫馨提示:
1: 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
2: 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權益歸上傳用戶所有。
3.本站RAR壓縮包中若帶圖紙,網頁內容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
4. 未經權益所有人同意不得將文件中的內容挪作商業(yè)或盈利用途。
5. 裝配圖網僅提供信息存儲空間,僅對用戶上傳內容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內容本身不做任何修改或編輯,并不能對任何下載內容負責。
6. 下載文件中如有侵權或不適當內容,請與我們聯(lián)系,我們立即糾正。
7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 川渝旅游日記成都重慶城市介紹推薦景點美食推薦
- XX國有企業(yè)黨委書記個人述責述廉報告及2025年重點工作計劃
- 世界濕地日濕地的含義及價值
- 20XX年春節(jié)節(jié)后復工安全生產培訓人到場心到崗
- 大唐女子圖鑒唐朝服飾之美器物之美繪畫之美生活之美
- 節(jié)后開工第一課輕松掌握各要點節(jié)后常見的八大危險
- 廈門城市旅游介紹廈門景點介紹廈門美食展示
- 節(jié)后開工第一課復工復產十注意節(jié)后復工十檢查
- 傳統(tǒng)文化百善孝為先孝道培訓
- 深圳城市旅游介紹景點推薦美食探索
- 節(jié)后復工安全生產培訓勿忘安全本心人人講安全個個會應急
- 預防性維修管理
- 常見閥門類型及特點
- 設備預防性維修
- 2.乳化液泵工理論考試試題含答案