数据结构课件(树和二叉树)_第1页
数据结构课件(树和二叉树)_第2页
数据结构课件(树和二叉树)_第3页
数据结构课件(树和二叉树)_第4页
数据结构课件(树和二叉树)_第5页
已阅读5页,还剩88页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1、6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 1数据结构数据结构第六章第六章 树和二叉树树和二叉树2021年年11月月24日星期三日星期三6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 2abcdefghijklm树树6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼

2、树与哈夫曼编码 36.1 6.1 树的类型定义树的类型定义数据对象数据对象d:d是具有相同特性的数据元素的集合。是具有相同特性的数据元素的集合。数据关系数据关系r: 若若d为空集,则称为空树;为空集,则称为空树; 否则否则:(1)在在d中存在唯一的称为根的数据元素中存在唯一的称为根的数据元素root,(2)当当n1时,其余结点可分为时,其余结点可分为m(m0)个互不相交的个互不相交的有限集有限集 t1, t2, , tm, 其中每一棵子集本身又是一棵符其中每一棵子集本身又是一棵符合本定义的树,称为根合本定义的树,称为根root的子树。的子树。基本操作:基本操作:查找:查找: root(t);

3、value(t, cur_e); parent(t, cur_e); leftchild(t, cur_e); treeempty(t); t r e e d e p t h ( t ) ; traversetree(t, visit( );6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 46.1 6.1 树的类型定义树的类型定义插入:插入: inittree(&t); createtree(&t, definition); assign(t, cur_e, va

4、lue); insertchild(&t, &p, i, c);删除:删除: cleartree(&t); destroytree(&t); deletechild(&t, &p, i); destroytree(&t);6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 56.1 6.1 树的类型定义树的类型定义和线性结构的比较和线性结构的比较 线性结构线性结构 树结构树结构第一个数据元素第一个数据元素(无前驱无前驱); 根结

5、点根结点(无前驱无前驱);最后一个数据元素最后一个数据元素(无后继无后继); 多个叶子结点多个叶子结点(无后继无后继);其它数据元素其它数据元素 树中其它结点树中其它结点(一个前驱、一个后继一个前驱、一个后继) 。 (一个前驱、多个后继一个前驱、多个后继)。6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 6abcdefghijklm树形表示树形表示a (b (e (k, l), f), c(g ), d( h( m), i, j)嵌套括号表示法嵌套括号表示法树的表示方法:树的表示

6、方法: i jfklgmccdbea文氏图文氏图凹入表凹入表abfeklcdhigmj6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 7基本术语基本术语结点结点: 数据元素数据元素 + 若干指向子树的分支。若干指向子树的分支。结点的度结点的度: 分支的个数。分支的个数。树的度树的度: 树中所有结点的度的最大值。树中所有结点的度的最大值。叶子结点叶子结点: 度为零的结点。度为零的结点。分支结点分支结点: 度大于零的结点;度大于零的结点; 路径路径和和路径长度路径长度。孩子孩子结点、

7、结点、双亲双亲结点、结点、兄弟兄弟结点、堂兄弟、结点、堂兄弟、祖先祖先结点、结点、子孙子孙结点。结点。边边:双亲节点:双亲节点x到子结点到子结点y的有序对的有序对(x,y)。结点的层次结点的层次: 假设根结点的层次为假设根结点的层次为1,第,第i 层的结点的层的结点的子树根结点的层次为子树根结点的层次为i+1。规定根的层数为规定根的层数为0。树的深度:树的深度:树中叶子结点所在的最大层次。树中叶子结点所在的最大层次。 森林:森林:是是m(m0)棵互不相交的树的集合任何一棵)棵互不相交的树的集合任何一棵非空树是一个二元组非空树是一个二元组tree = (root,f)其中:其中:root被称为根

8、结点,被称为根结点,f被称为子树森林。被称为子树森林。6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 86.2 6.2 二叉树的类型定义二叉树的类型定义二叉树的定义二叉树的定义定义:定义:二叉树是由二叉树是由n(n=0)个结点的有限集合构个结点的有限集合构成,此集合或者为空集,或者由一个根结点及两棵互成,此集合或者为空集,或者由一个根结点及两棵互不相交的左右子树组成,并且左右子树都是二叉树。不相交的左右子树组成,并且左右子树都是二叉树。与树的关系:与树的关系:这也是一个递归定义。

9、这也是一个递归定义。区别区别: 二叉树结点的子树要区分左子树和右二叉树结点的子树要区分左子树和右子树,即使只有一棵子树也要进行区分,说明它是左子树,即使只有一棵子树也要进行区分,说明它是左子树,还是右子树。子树,还是右子树。6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 9(a)空二叉树空二叉树a (b)根和空的根和空的左右子树左右子树ab (c)根和左子树根和左子树二叉树的二叉树的5 5种基本形态种基本形态a(d)根和右子树根和右子树ba (e)根和左右子树根和左右子树bc6.

10、1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 10二叉树的主要基本操作:二叉树的主要基本操作:查找:查找: root(t); value(t, e);parent(t, e); leftchild(t, e); rightchild(t, e); leftsibling(t, e); rightsibling(t, e); bitreeempty(t); bitreedepth(t); preordertraverse(t, visit(); inordertraverse(t,

11、visit(); postordertraverse(t, visit(); levelordertraverse(t, visit();插入:插入: initbitree(&t); assign(t, &e, value); createbitree(&t, definition); insertchild(t, p, lr, c);删除:删除: clearbitree(&t); destroybitree(&t); deletechild(t, p, lr); 6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的

12、遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 11性质性质1: 在二叉树的第在二叉树的第i层上至多有层上至多有2i-1个结点个结点(i=1)。 证明:证明:采用归纳法证明此性质。采用归纳法证明此性质。当当i=1时,只有一个根结点,时,只有一个根结点,2i-1=20 =1,命题成立。命题成立。现在假定第现在假定第i1层上命题成立,则第层上命题成立,则第i1层上至层上至多有多有2i-2个结点。由于二叉树每个结点的度最大为个结点。由于二叉树每个结点的度最大为2,故在第故在第i层上最大结点数为第层上最大结点数为第i1层上最大结点数的二层上最大结点数的二倍,倍, 即即22i-

13、22i-1。命题得到证明。命题得到证明。二叉树的重要特性:二叉树的重要特性:6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 12性质性质2:深度为:深度为k的二叉树至多有的二叉树至多有2k1个结点(个结点(k=1).证明:证明:深度为深度为k的二叉树的最大的结点时为二叉的二叉树的最大的结点时为二叉树中每层上的最大结点数之和,由性质树中每层上的最大结点数之和,由性质1得到每层上的得到每层上的最大结点数最大结点数2i-1(第(第i层上的最大结点数)层上的最大结点数)二叉树的重要特性:

14、二叉树的重要特性:12k1i1i2k6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 13二叉树的重要特性:二叉树的重要特性:性质性质3: 对任何一棵二叉树,如果其终端结点数为对任何一棵二叉树,如果其终端结点数为n0,度为度为2的结点数为的结点数为n2,则则n0n21。证明:证明:设二叉树中度为设二叉树中度为1的结点数为的结点数为n1,二叉树中总结点,二叉树中总结点数为数为n,因为二叉树中所有结点均小于或等于,因为二叉树中所有结点均小于或等于2,所以有:,所以有:nn0n1n2 (

15、1) 再看二叉树中的分支数,除根结点外,其余结点都再看二叉树中的分支数,除根结点外,其余结点都有一个进入分支,设有一个进入分支,设b为二叉树中的分支总数,为二叉树中的分支总数, 则有:则有:nb1。由于这些分支都是由度为。由于这些分支都是由度为1和和2的结点射出的,的结点射出的,所以有:所以有:bn1+2*n2 nb1n12n21 (2)由式(由式(1)和()和(2)得到:)得到: n0+n1+n2=n1+2*n2+1 n0n216.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 1

16、4满二叉树满二叉树2453671123456(a)完全二叉树完全二叉树123457(b)非完全二叉树非完全二叉树12367( c)非完全二叉树非完全二叉树6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 15两种特殊形态的二叉树:两种特殊形态的二叉树: 一棵深度为一棵深度为k且由且由2k-1个结点的二叉树称为个结点的二叉树称为满二叉树满二叉树。如果深度为如果深度为k、由、由n个结点的二叉树中各结点能够与个结点的二叉树中各结点能够与深度为深度为k的顺序编号的满二叉树从的顺序编号的满二

17、叉树从1到到n标号的结点相对标号的结点相对应,则称这样的二叉树为应,则称这样的二叉树为完全二叉树完全二叉树。完全二叉树的特点完全二叉树的特点是:是:(1)所有的叶结点都出现在第)所有的叶结点都出现在第k层或层或k1层。层。(2)对任一结点,如果其右子树的最大层次为)对任一结点,如果其右子树的最大层次为h,则其左子树的最大层次为则其左子树的最大层次为h或或h1。满二叉树和完全二叉树满二叉树和完全二叉树6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 16 性质性质4:具有:具有n个结

18、点的完全二叉树的深度为个结点的完全二叉树的深度为log2n1。符号符号x表示不大于表示不大于x的最大整数。的最大整数。 证明:证明:假设此二叉树的深度为假设此二叉树的深度为k,则根据性质,则根据性质2及完全及完全二叉树的定义得到:二叉树的定义得到:2k-11n=2k-1 或或 2k-1=n2k取对数得到:取对数得到:k1log2nk 因为因为k是整数。所以有:是整数。所以有:klog2n1。二叉树的重要特性二叉树的重要特性6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 17性质性

19、质5: 如果对一棵有如果对一棵有n个结点的完全二叉树的结点按个结点的完全二叉树的结点按层序编号(从第层序编号(从第1层到第层到第log2n+1层,每层从左到右层,每层从左到右),则则对任一结点对任一结点i(1=i1,则其双亲是结点,则其双亲是结点i/2。 (2)如果)如果2in,则结点,则结点i为叶子结点,无左孩子;否为叶子结点,无左孩子;否则,其左孩子是结点则,其左孩子是结点2i。 (3)如果)如果2i1n,则结点,则结点i无右孩子;否则,其右孩无右孩子;否则,其右孩子是结点子是结点2i1。证明:略!证明:略!完全二叉树的特点完全二叉树的特点: :6.1 树的类型定义 6.2 二叉树的类型定

20、义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 186.3 6.3 二叉树的存储结构二叉树的存储结构1.顺序存储结构顺序存储结构它是用一组连续的存储单元存储二叉树的数据元素。它是用一组连续的存储单元存储二叉树的数据元素。因此,必须把二叉树的所有结点安排成为一个恰当的序因此,必须把二叉树的所有结点安排成为一个恰当的序列,结点在这个序列中的相互位置能反映出结点之间的列,结点在这个序列中的相互位置能反映出结点之间的逻辑关系,用编号的方法:逻辑关系,用编号的方法: #define max-tree-size 100typedef

21、telemtype sqbitreemax-tree-size;sqbitree bt 从树根起,自上层至下层,每层自左至右的给所有从树根起,自上层至下层,每层自左至右的给所有结点编号。结点编号。6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 19lkjihgfedcba 1 2 3 4 5 6 7 8 9 10 11 12完全二叉树完全二叉树abcdefghijkl6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树

22、 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 20abcdefg 表示该处没有元素表示该处没有元素存在仅仅为了好理解存在仅仅为了好理解abcdefg一般二叉树一般二叉树6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 211.1.顺序存储结构顺序存储结构缺点:缺点:有可能对存储空间造成极大的浪费,在最有可能对存储空间造成极大的浪费,在最坏的情况下,一个深度为坏的情况下,一个深度为h且只有且只有h个结点的个结点的右单支树右单支树确需要确需要2h-1个结点存储空间。而且,若经常需要

23、插入与个结点存储空间。而且,若经常需要插入与删除树中结点时,顺序存储方式不是很好!删除树中结点时,顺序存储方式不是很好!6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 221.1.顺序存储结构顺序存储结构abcdindexindex 0 01 1 2 23 34 4 5 5 6 6 7 78 8 9 9 1010 1111 1212 1313 1414 1515a a b b c c d d- - - -6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6

24、.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 23lchilddatarchildabcdefgh(2 2)二叉链表法)二叉链表法abcdefgh6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 24typedef struct bitnode telemtype data; struct bitnode *lchild,*rchild; bitnode,*bitree;2 2)二叉链表法)二叉链表法二叉树的二叉链表存储表示二叉树的二叉链表

25、存储表示6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 25status createbitree(bitree *t) /*创建一棵二叉树创建一棵二叉树*/ scanf(&ch); if(ch= =) t=null; else if(!(t=(bitnode *)malloc(sizeof(bitnode) exit(overflow); tdata=ch; createbitree(tlchild); createbitree(trchildd); return ok

26、; 6.3 6.3 二叉树的存储结构二叉树的存储结构链式存储结构的算法:链式存储结构的算法:6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 26三叉链表法三叉链表法abcdefghdbcefgharchildparentdatalchild6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 27typedef struct tbitnode telemtype data

27、; struct tbitnode *lchild,*rchild,*parent; bitnode,*bitree;三叉链表法三叉链表法二叉树的三叉链表存储表示二叉树的三叉链表存储表示6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 286.4 6.4 二叉树的遍历二叉树的遍历一、问题的提出一、问题的提出顺着某一条搜索路径巡访二叉树中的结点,使得每顺着某一条搜索路径巡访二叉树中的结点,使得每个结点均被访问一次,而且仅被访问一次。个结点均被访问一次,而且仅被访问一次。“访问访问”的

28、含义可以很广,如:输出结点的信息的含义可以很广,如:输出结点的信息等。等。 对对“二叉树二叉树”而言,可以有三条搜索路径:而言,可以有三条搜索路径:1先上后下的按层次遍历;先上后下的按层次遍历;2先左(子树)后右(子树)的遍历;先左(子树)后右(子树)的遍历;3先右(子树)后左(子树)的遍历。先右(子树)后左(子树)的遍历。6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 296.4 6.4 二叉树的遍历二叉树的遍历二、先左后右的遍历算法二、先左后右的遍历算法 先(根)序的遍历算法

29、:先(根)序的遍历算法:若二叉树为空树,则空操作;否则,(若二叉树为空树,则空操作;否则,(1)访问根结点;)访问根结点; (2)先序遍历左子树;)先序遍历左子树; (3)先序遍历右子树。)先序遍历右子树。中(根)序的遍历算法:中(根)序的遍历算法:若二叉树为空树若二叉树为空树,则空操作;否则则空操作;否则,(1)中序遍历左子树;)中序遍历左子树;(2)访问根结点;)访问根结点; (3)中序遍历右子树。)中序遍历右子树。后(根)序的遍历算法:后(根)序的遍历算法:若二叉树为空树若二叉树为空树,则空操作则空操作;否则否则,(1)后序遍历左子树;)后序遍历左子树; (2)后序遍历右子树;)后序遍历

30、右子树; (3)访问根结点。)访问根结点。 6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 306.4 6.4 二叉树的遍历二叉树的遍历示例示例/-abcd图图1 (a-b)/(c+d)/bca-+d图图2 a-(b/c+d)/bca-+d图图3 a-b/c+d先序:先序:/-ab+cd中序:中序:a-b/c+d后序:后序:ab-cd+/先序:先序:-a+/bcd中序:中序:a-b/c+d后序:后序:abc/d+-先序:先序:+-a/bcd中序:中序:a-b/c+d后序:后序:a

31、bc/-d+分别称为表达式的前缀表示法、中缀表示法和后缀表示分别称为表达式的前缀表示法、中缀表示法和后缀表示法(逆波兰表示法)。法(逆波兰表示法)。6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 316.4 6.4 二叉树的遍历二叉树的遍历三、算法的递归描述三、算法的递归描述 void preorder (bitree t, void( *visit)(telemtype& e) / 先序遍历二叉树先序遍历二叉树 if (t) visit(t-data); / 访问结点访

32、问结点preorder(t-lchild, visit); / 遍历左子树遍历左子树preorder(t-rchild, visit); / 遍历右子树遍历右子树6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 32void inordertraverse (bitree t, void( *visit)(telemtype &e)/中序遍历中序遍历if(t)inordertraverse (t-lchild, visit); visit(t-data); / 访问结点访问

33、结点inordertraverse (t-rchild, visit);void postordertraverse (bitree t, void( *visit)(telemtype &e)/后序遍历后序遍历if(t)postordertraverse (t-lchild, visit);postordertraverse (t-rchild, visit); visit(t-data); / 访问结点访问结点6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 33四、先

34、序遍历算法的非递归描述四、先序遍历算法的非递归描述先序遍历的非递归算法。先序遍历的非递归算法。t指向根,指向根,p为指针,指为指针,指向当前结点。使用一个堆栈向当前结点。使用一个堆栈s(),(),top为栈指针为栈指针 1 if t=null 2 then 输出输出 这是一棵空树这是一棵空树 3 else p=t,top=0 4 do 5 while p!=null 6 输出输出 data(p); 7 top+;8 if topn9 then 调用调用 栈满栈满 10 else stop=p,p=lchild(p)11 if top!=012 p=stop; top-; p=rchild(p)

35、13while( top=0 & p=null)算法结束算法结束6.4 6.4 二叉树的遍历二叉树的遍历四、先序遍历算法的非递归描述四、先序遍历算法的非递归描述1 if t=null2 then 输出输出 这是一棵空树这是一棵空树 3 else p=t,top=0 4 do 5 while p!=null 6 输出输出 data(p);7 top+;8 if topn9 then 调用调用 栈满栈满10 else stop=p,p=lchild(p)11 if top!=012 p=stop; top-; p=rchild(p)13while( top=0 & p=null)算

36、法结束算法结束注注:结点旁结点旁的数字是的数字是该结点的该结点的存储地址存储地址二叉树执行先序遍历二叉树执行先序遍历算法的过程算法的过程栈栈6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 35中序遍历的非递归算法中序遍历的非递归算法中序遍历的非递归算法,使用堆栈中序遍历的非递归算法,使用堆栈s(),(),top为指针,为指针,t指向根,指向根,p为指针,指向当前结点为指针,指向当前结点1 top=0,p=t2 do 3 while ( p!=nil ) 4 top+ 5 if (

37、topn )6 then 调用调用 栈满栈满7 else stop=p; p=lchild(p)8 if top!=09 then p=stop, top-10 输出输出 data(p) 11 pn )6 then 调用调用 栈满栈满7 else stop=p; p=lchild(p)8 if top!=09 then p=stop, top-10 输出输出 data(p) 11 plchild)& (!t-rchild)count+;countleaf( t-lchild, count); / 统计左子树中叶子结点个数统计左子树中叶子结点个数countleaf( t-rchild,

38、count);/ 统计右子树中叶子结点个数统计右子树中叶子结点个数6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 38五、遍历算法的应用举例五、遍历算法的应用举例2、求二叉树的深度、求二叉树的深度(后序遍历后序遍历)int depth (bitree t ) if ( !t ) depthval = 0; else depthleft = depth( t-lchild );depthright= depth( t-rchild );depthval = 1+(depthleft

39、 depthright? depthleft:depthright); return depthval; 6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 39五、遍历算法的应用举例五、遍历算法的应用举例3、复制二叉树、复制二叉树(后序遍历后序遍历)/ 生成一个二叉树的结点生成一个二叉树的结点bitnode *gettreenode(telemtype item,bitnode *lptr , bitnode *rptr ) if (!(t = (bitnode*)malloc(s

40、izeof(bitnode)exit(1); t- data = item; t- lchild = lptr; t- rchild = rptr; return t;6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 40五、遍历算法的应用举例五、遍历算法的应用举例3、复制二叉树、复制二叉树(后序遍历后序遍历)bitnode *copytree(bitnode *t) if (!t )return null;if (t-lchild ) newlptr = copytree(t-l

41、child);else newlptr = null;if (t-rchild ) newrptr = copytree(t-rchild);else newrptr = null;newnode = gettreenode(t-data, newlptr, newrptr);return newnode;6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 416.5 6.5 线索二叉树线索二叉树一、何谓线索二叉树?一、何谓线索二叉树?遍历二叉树是按一定的规则将二叉树中结点排列成遍历

42、二叉树是按一定的规则将二叉树中结点排列成一个线性序列;这实际上是把一个非线性结构进行线性一个线性序列;这实际上是把一个非线性结构进行线性化的操作。化的操作。以二叉链表作为存储结构时,对于某个结点只能找以二叉链表作为存储结构时,对于某个结点只能找到其左右孩子,而不能直接得到结点在任一序列中的前到其左右孩子,而不能直接得到结点在任一序列中的前驱或后继。要想得到只能通过遍历的动态过程才行。驱或后继。要想得到只能通过遍历的动态过程才行。怎样保存遍历过程中得到的信息呢?怎样保存遍历过程中得到的信息呢?(1增加两个指针增加两个指针(2利用结构中的空链域,并设立标志。即采用如利用结构中的空链域,并设立标志。

43、即采用如下的形式:下的形式:lchild ltagdatartagrchild6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 426.5 6.5 线索二叉树线索二叉树何谓线索二叉树?何谓线索二叉树? 指向该线性序列中的指向该线性序列中的“前驱前驱”和和“后继后继”的的指针指针,称,称作作“线索线索”。包含。包含“线索线索”的存储结构,称作的存储结构,称作“线索链线索链表表”;与其相应的二叉树,称作;与其相应的二叉树,称作“线索二叉树线索二叉树”。对线索链表中结点的约定:对线索链表

44、中结点的约定:在二叉链表的结点中增加两个标志域,并作如下规定:在二叉链表的结点中增加两个标志域,并作如下规定:若该结点的左子树不空,则若该结点的左子树不空,则lchild域的指针指向其域的指针指向其左子左子树树,且左标志域,且左标志域 ltag的值为的值为0; 否则,否则,lchild域的指针指向其域的指针指向其“前驱前驱”,且左标志,且左标志ltag的值的值为为1。若该结点的右子树不空,则若该结点的右子树不空,则rchild域的指针指向其右域的指针指向其右子树,且右标志域子树,且右标志域rtag的值为的值为0;否则,否则,rchild域的指针指向其域的指针指向其“后继后继”,且右标志,且右标

45、志rtag的值为的值为1。 6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 436.5 6.5 线索二叉树线索二叉树0 a 01b 00d 1 1c 11e 1tadbec中序序列:中序序列:bcaed6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 446.5 6.5 线索二叉树线索二叉树线索链表的结构描述:线索链表的结构描述:typedef enum link,

46、thread pointerthr; / link=0:指针,指针,thread=1:线索线索typedef struct bithrnodetelemtype data;struct bithrnode *lchild, *rchild; / 左右指针左右指针pointerthr ltag, rtag; / 左右标志左右标志 bithrnode, *bithrtree;6.5 6.5 线索二叉树线索二叉树找结点的后继找结点的后继线索化线索化:对二叉树以某种次序遍历使其变为线索二叉树的:对二叉树以某种次序遍历使其变为线索二叉树的过程叫做过程叫做线索化线索化。下面以下面以中序中序为例看看如何在线

47、索二叉树中为例看看如何在线索二叉树中找结点的后继找结点的后继。(1树中所有叶子结点和只有左子树的右指针均为线索,树中所有叶子结点和只有左子树的右指针均为线索,直接指示了结点的后继;直接指示了结点的后继;(2其他非终端结点的右链均为指针,无法得到后继的信其他非终端结点的右链均为指针,无法得到后继的信息。然而根据中序遍历的规律可知,结点的后继应是遍历其右息。然而根据中序遍历的规律可知,结点的后继应是遍历其右子树时访问的第一个结点,即右子树中最左下的结点。子树时访问的第一个结点,即右子树中最左下的结点。(3反之,在中序线索树中找结点前驱的规律是:若左标反之,在中序线索树中找结点前驱的规律是:若左标志

48、是志是1,则左链为线索,指示其前驱;否则,遍历该结点的左子,则左链为线索,指示其前驱;否则,遍历该结点的左子树时最后访问的结点(左子树中最右下的结点为其前驱树时最后访问的结点(左子树中最右下的结点为其前驱)。abceidhf ggknullnull6.5 6.5 线索二叉树线索二叉树找结点的后继找结点的后继在在后序线索树后序线索树中找结点中找结点x的后继较复杂,可分三种情况:的后继较复杂,可分三种情况:(1若结点若结点x是二叉树的根是二叉树的根,则其后继为空则其后继为空;(2若结点若结点x是其双亲的右孩子或是左孩子且其双亲没有是其双亲的右孩子或是左孩子且其双亲没有右子树右子树,则其后继即为双亲

49、结点。则其后继即为双亲结点。(3若结点若结点x是其双亲的左孩子,且其双亲有右子树,则是其双亲的左孩子,且其双亲有右子树,则其后继为双亲的右子树上按后序遍历出的第一个结点。其后继为双亲的右子树上按后序遍历出的第一个结点。由此可见,在后序线索化树上找后继时需知道结点的双亲,由此可见,在后序线索化树上找后继时需知道结点的双亲,因此需要使用三叉链表。因此需要使用三叉链表。可见,在中序线索二叉树上遍历二叉树,虽然时间复杂度可见,在中序线索二叉树上遍历二叉树,虽然时间复杂度也是也是o(n),但常数因子小,且不需要设栈,因此若在某程序中,但常数因子小,且不需要设栈,因此若在某程序中需要经常遍历或查找结点在遍

50、历所得线性序列中的前驱和后继,需要经常遍历或查找结点在遍历所得线性序列中的前驱和后继,则应采用线索链表作存储结构。则应采用线索链表作存储结构。6.5 6.5 线索二叉树线索二叉树二、线索链表的遍历算法二、线索链表的遍历算法:for ( p = firstnode(t); p; p = succ(p) )visit(p);中序线索化链表的遍历算法中序线索化链表的遍历算法:中序遍历的第一个结点中序遍历的第一个结点 ?在中序线索化链表中结点的后继在中序线索化链表中结点的后继 ?abceidhf ggknullnull6.5 6.5 线索二叉树线索二叉树二、线索链表的遍历算法二、线索链表的遍历算法:s

51、tatus inordertraverse_thr(bithrtree t, status (*visit)(telemtype e) / t指向头结点,头指向头结点,头结点的左链结点的左链lchild指向根结点,头结点的右链指向根结点,头结点的右链rchild, 指向指向中序遍历的最后一个结点。中序遍历二叉线索链表表示的中序遍历的最后一个结点。中序遍历二叉线索链表表示的二叉树,对每个数据元素调用函数二叉树,对每个数据元素调用函数visit。p = t-lchild; / p指向根结点指向根结点while (p != t) / 空树或遍历结束时,空树或遍历结束时,p=t while (p-lt

52、ag=link) p = p-lchild; if (!visit(p-data) return error;/ 访问其左访问其左子树为空的结点子树为空的结点 while (p-rtag=thread & p-rchild!=t) p = p-rchild; visit(p-data); / 访问后继结点访问后继结点 p = p-rchild; / p进至其右子树根进至其右子树根 return ok; / inordertraverse_thrt-+/*-abe fc d6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树

53、6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 496.5 6.5 线索二叉树线索二叉树三、如何建立线索链表?三、如何建立线索链表?在中序遍历过程中修改结点的左、右指针域,在中序遍历过程中修改结点的左、右指针域,以保存当前访问结点的以保存当前访问结点的“前驱前驱”和和“后继后继”信息。信息。遍历过程中,附设指针遍历过程中,附设指针pre,并始终保持指针并始终保持指针pre指向当前访问的、指针指向当前访问的、指针p所指结点的前驱。所指结点的前驱。6.5 6.5 线索二叉树线索二叉树三、如何建立线索链表?三、如何建立线索链表?status inorderthreading(bithrtree &a

54、mp;thrt, bithrtree t) / 中中序遍历二叉树序遍历二叉树t,并将其中序线索化,并将其中序线索化,thrt指向头结点。指向头结点。if (!(thrt = (bithrtree)malloc(sizeof(bithrnode) exit (overflow);thrt-ltag = link; thrt-rtag =thread; / 建头结点建头结点thrt-rchild = thrt; / 右指针回指右指针回指if (!t) thrt-lchild = thrt; / 若二叉树空,则左指针回指若二叉树空,则左指针回指else thrt-lchild = t; pre =

55、thrt;inthreading(t); / 中序遍历进行中序线索化中序遍历进行中序线索化pre-rchild = thrt; pre-rtag = thread; / 最后一个结点线索化最后一个结点线索化thrt-rchild = pre; return ok; / inorderthreadingvoid inthreading(bithrtree p) if (p) inthreading(p-lchild); / 左子树线索化左子树线索化if (!p-lchild) p-ltag = thread; p-lchild = pre; / 建前驱线索建前驱线索if (!pre-rchild

56、) pre-rtag = thread; pre-rchild = p; / 建后继线索建后继线索pre = p; / 保持保持pre指向指向p的前驱的前驱inthreading(p-rchild); / 右子树线索化右子树线索化 / inthreading6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 526.6 6.6 树和森林树和森林一、树的三种存储结构一、树的三种存储结构1. 双亲表示法双亲表示法:在这种表示中,容易找到父结点及其所有的祖先;在这种表示中,容易找到父结点及

57、其所有的祖先;也能找到结点的子女和兄弟(虽然比较麻烦)。但它没有也能找到结点的子女和兄弟(虽然比较麻烦)。但它没有表示出结点之间的左右次序,所以表示出结点之间的左右次序,所以无法求树中某个指定结无法求树中某个指定结点的最左子结点和右兄弟结点等基本运算点的最左子结点和右兄弟结点等基本运算。abcdefghijklm8m5l5k4j4i4h3g2f2e1d1c1b-1a123456789101112136.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 53一、一、树的三种存储结构树的三

58、种存储结构1. 双亲表示法双亲表示法:用一组连续空间存储树的结点,并附设一个指用一组连续空间存储树的结点,并附设一个指示器指示其双亲结点的位置。结构类型如下:示器指示其双亲结点的位置。结构类型如下:#define max_tree_size 100结点结构结点结构:typedef struct ptnode /* 树中结点结构树中结点结构 */elem data; /* 结点中的元素结点中的元素 */int parent; / 双亲位置域双亲位置域 ptnode;树结构树结构:typedef struct ptnode nodesmax_tree_size;int r, n; / 根结点的位置

59、和结点个数根结点的位置和结点个数 ptree一、树的三种存储结构一、树的三种存储结构2. 孩子链表表示法孩子链表表示法:abcdefghijklmmlkjihgfedcba12345678910111213234 56 13 78910 1112 6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉树 6.6 树和森林 6.7 哈夫曼树与哈夫曼编码 55一、树的三种存储结构一、树的三种存储结构2、孩子链表表示法、孩子链表表示法:结点表中的每一元素又包含一个子表,存放该结点结点表中的每一元素又包含一个子表,存放该结点的所有子结点。结点表

60、顺序存放,子表用链接表示,子的所有子结点。结点表顺序存放,子表用链接表示,子表中的元素按从左到右的次序存放。结构类型如下:表中的元素按从左到右的次序存放。结构类型如下:双亲结点结构双亲结点结构typedef struct elem data;childptr firstchild; / 孩子链的头指针孩子链的头指针 ctbox;树结构树结构:typedef struct ctbox nodesmax_tree_size;int n, r; / 结点数和根结点的位置结点数和根结点的位置 ctree;孩子结点结构孩子结点结构:typedef struct ctnode int child;struct ctnode *next; *childptr;6.1 树的类型定义 6.2 二叉树的类型定义 6.3 二叉树的存储结构 6.4 二叉树的遍历 6.5 线索二叉

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论