【数据结构】一篇深入理解二叉树计算

简介: 【数据结构】一篇深入理解二叉树计算

I.树的概念及结构

树的概念

 树是一种非线性的数据结构,它是由 n(n >= 0)个有限节点组成的一个具有层次关系的集合。

   那么为什么叫 "树" 呢?

   我们之所以把它成为 "树",是因为它很像我们现实生活中的树。只是它是倒过来的,根朝上叶子朝下。

树的结构

特点:


有一个特殊的结点,称为根结点,根节点没有前驱结点

除根节点外,其余结点被分成M(M>0)个互不相交的集合T1、T2、……、Tm,其中每一个集合Ti(1<= i<= m)又是一棵结构与树类似的子树。每棵子树的根结点有且只有一个前驱,可以有0个或多个后继

因此,树是递归定义的。


注意:

  1. 树形结构中,子树之间不能有交集,否则就不是树形结构。
  2. 除了根结点外,每个结点有且仅有一个父结点。
  3. 一颗N个结点的树由N-1条边。

树的专有名词

专有名词:

节点的度:一个节点含有的子树的个数称为该节点的度; 如上图:A的为6

叶节点或终端节点:度为0的节点称为叶节点; 如上图:B、C、H、I...等节点为叶节点

非终端节点或分支节点:度不为0的节点; 如上图:D、E、F、G...等节点为分支节点

双亲节点或父节点:若一个节点含有子节点,则这个节点称为其子节点的父节点; 如上图:A是B的父节点

孩子节点或子节点:一个节点含有的子树的根节点称为该节点的子节点; 如上图:B是A的孩子节点

兄弟节点:具有相同父节点的节点互称为兄弟节点; 如上图:B、C是兄弟节点

树的度:一棵树中,最大的节点的度称为树的度; 如上图:树的度为6

节点的层次:从根开始定义起,根为第1层,根的子节点为第2层,以此类推;

树的高度或深度:树中节点的最大层次; 如上图:树的高度为4

堂兄弟节点:双亲在同一层的节点互为堂兄弟;如上图:H、I互为兄弟节点

节点的祖先:从根到该节点所经分支上的所有节点;如上图:A是所有节点的祖先

子孙:以某节点为根的子树中任一节点都称为该节点的子孙。如上图:所有节点都是A的子孙

森林:由m(m>0)棵互不相交的树的集合称为森林;

最近公共祖先:距离某些结点最近的祖先,比如P和Q的最近公共祖先为J,K和F的最近公共祖先为F。结点本身也可以成为自己的祖先

 树的表示

以前学单链表时只有一个指针,双链表两个指针,但是树有多少个指针是不确定的,因为树没有规定一个节点最多有多少个孩子。那我们该如何定义结构呢?


如若知道树的度,那就很好定义了:

//假设指定树的度,可以直接定义
#define N 5
struct TreeNode
{
  int data;
  struct TreeNode* subs[N]; //指针数组,每个结点存5个
};

但是又有一个缺陷,既然树的度为5,但是你这样设定只会导致每个结点的度均为5,可实际上只要保证每个结点的度<=5即可,由此可见,此法开辟造成空间浪费。此外,如果不知道树的度,那么用上述方法定义就比较困难了


问题点:

① 可能会存在不少的空间浪费。

② 万一没有限定树的度为多少呢?这个方式就废了。


树的表示最佳方法:孩子兄弟表示法

typedef int DataType;
struct Node {
    struct Node* _firstChind1;   // 永远指向第一个孩子
    struct Node* _pNextBrother;  // 指向孩子右边的兄弟
    DataType _data;
};

既要保存值域,也要保存结点和结点之间的关系,实际中树有很多种表示方式如:双亲表示法,孩子表示法、孩子双亲表示法以及孩子兄弟表示法等。我们这里就简单的了解其中最常用的孩子兄弟表示法。

假设我们要表示如下的树就可以如下图注意理解

物理结构如下:

原理:


由上图代码图画演示及树的样子得知,根结点A是有B和C两个孩子的,永远让A的leftchild指向第一个孩子B,A的其它孩子C让B的兄弟指针去指向,C没有兄弟了,直接指向NULL。同理,B有3个孩子,让leftchild指针指向第一个孩子D,再让D的兄弟指针指向下一个E,以此类推……


解读:

无论你有多少个孩子,它都只存两个指针。一个指针永远指向第一个孩子,另一个指针指向孩子右边的兄弟(亲兄弟)。这个树的度无论为多少,也不需要用顺序表存,但是你任何一个节点有多少个孩子都能给你表示出来,通过第一个孩子把所有孩子都找出来。不复杂也没有浪费,只用两个指针就把链接关系都表示出来了


树在实际中的运用

文件系统的目录树结构、网络拓扑,最短路径问题,搜索引擎、思维导图等

II.二叉树的概念及结构

二叉树的概念

二叉树:度为2的树,一棵二叉树是结点的一个有限集合,该集合:

  1. 或者为空
  2. 由一个根节点加上两棵别称为左子树和右子树的二叉树组成

从上图可以看出:

  1. 二叉树不存在度大于2的结点
  2. 二叉树的子树有左右之分,次序不能颠倒,因此二叉树是有序树

注意:对于任意的二叉树都是由以下几种情况复合而成的:

 特殊的二叉树

满二叉树:一个二叉树,如果每一个层的结点数都达到最大值,则这个二叉树就是满二叉树。也就是说,如果一个二叉树的层数为K,且结点总数是 2^k -1 ,则它就是满二叉树。


完全二叉树:完全二叉树是效率很高的数据结构,完全二叉树是由满二叉树而引出来的。对于深度为K的,有n个结点的二叉树,当且仅当其每一个结点都与深度为K的满二叉树中编号从1至n的结点一一对应时称之为完全二叉树。 要注意的是满二叉树是一种特殊的完全二叉树。


简单来说,满二叉树的每个结点的度均为2,满二叉树是完全二叉树的特殊情况。当深度为K时,完全二叉树就是在【1,k-1】层的区间内均为满二叉树,只有最后一层第K层不满,但是最后一层是从左往右连续的。图示:

二叉树的性质

性质1:若规定根节点的层数为1,则一棵非空二叉树的第i层上最多有2^(i-1) 个结点.


性质2:若规定根节点的层数为1,则深度为h的二叉树的最大结点数是2^h - 1 .


性质3:对任何一棵非空二叉树, 如果度为0其叶结点个数为n0 , 度为2的分支结点个数为n2 ,则有n0 =n2 +1


性质4:若规定根节点的层数为1,具有N个结点的满二叉树的深度,h= log2(N+1) .


性质5:对于具有n个结点的完全二叉树,如果按照从上至下从左至右的数组顺序对所有节点从0开始编号,则对于序号为i的结点有:


若i>0,i位置节点的双亲序号:(i-1)/2;i=0,i为根节点编号,无双亲节点

若2i+1<n,左孩子序号:2i+1,2i+1>=n否则无左孩子

若2i+2<n,右孩子序号:2i+2,2i+2>=n否则无右孩子


满二叉树计算公式:

  • 已知层数求总数:
  • 已知总数求层数:

完全二叉树小知识:

  • 完全二叉树中,度为 1 的最多只有 1 个。
  • 高度为 的完全二叉树节点范围是    

对于性质5计算左右孩子:

假设 是父节点在数组中的下标,此公式仅适用于完全二叉树:

① 求左孩子:

② 求右孩子:

③ 求父亲(假设不关注是左孩子还是右孩子):

④  判断是否有左孩子:    

⑤  判断是否由右孩子:    

解析:

由图中不难发现,所有左孩子的下标均为奇数,而右孩子下标均为偶数,所以不难得出左右孩子的表达式。相反可以推出父亲的表达式。

  • 但是父亲的式子中,只是(child - 1) / 2,并没有区分左右孩子,这是为什么呢?


这里我们假设leftchild和rightchild的下标均为3,算出来的parent下标不难发现均为1,同理左右孩子下标均为4时,父亲的下标算出来都是1,由此规律可得知直接用child下标表示父亲即可,无需区分左右孩子。

 

III.例题巩固

1. 某二叉树共有 399 个结点,其中有 199 个度为 2 的结点,则该二叉树中的叶子结点数为( )

    A.  不存在这样的二叉树

    B.  200

    C.  198

    D.  199


解:

由性质可知n0总是比n2度为2的结点多1,所以n0=200,选B


2. 在具有 2n 个结点的完全二叉树中,叶子结点个数为( )

    A.  n

    B.  n+1

    C.  n-1

    D.  n/2


解:

设度为i的节点个数为ni,那么结点总个数就是这些结点的总和,即n=n0+n1+n2+n3+……+n(i-1)+n(i)

所以总节点数2n=n0+n1+n2

由性质n0-1=n2,则式子变为n0-1+n1+n0=2n

又因为是完全二叉树,n不可能出现小数,所以n1=1

综上,n0=n,选A


3.  一棵完全二叉树的节点数位为531个,那么这棵树的高度为( )

    A.  11

    B.  10

    C.  8

    D.  12


解:

已知层数求总数:

1024=2^10;

512=2^9;

易知h大于9层不足10层,选B


5.  一个具有767个节点的完全二叉树,其叶子节点个数为()

    A.  383

    B.  384

    C.  385

    D.  386


解:

n=n0+n1+n2

n0-1=n2

因完全二叉树,n1只能为0或1,又要确保结果是整数,所以n1=0

选B


6.  在一颗度为3的树中,度为3的结点有2个,度为2的结点有1个,度为1的结点有2个,则叶子结点有()个

      A.  4

      B.  5

      C.  6

      D.  7


解:

由结点总数和边数的关系:结点总数是n,那么边数就是n-1;

设度为i的节点个数为ni,那么结点总个数就是这些结点的总和,即n=n0+n1+n2+n3+……+n(i-1)+n(i)

因为是度为3的数:n=n0+n1+n2+n3

总边数与度之间的关系为n-1=0*n0+1*n1+2*n2+3*n3

根据题意:n3=2,n2=1,n1=2

联立两个方程求解,可以得到n0 = n2 + 2n3 + 1, n0=6

所以选C


相关文章
|
2天前
【数据结构】二叉树(遍历,递归)
【数据结构】二叉树(遍历,递归
16 2
|
2天前
|
数据可视化
数据结构——lesson8二叉树的实现
本文介绍了二叉树的基本操作和实现,包括二叉树的构建、销毁、节点个数计算、叶子节点个数、第k层节点个数、查找、高度计算以及判断是否为完全二叉树的方法。通过递归和层序遍历等技巧,详细阐述了这些操作的原理和代码实现。文章以实例和图解帮助读者理解二叉树的各种特性和操作。
|
2天前
|
算法 编译器 C语言
数据结构——二叉树四种遍历的实现-3
数据结构——二叉树四种遍历的实现
数据结构——二叉树四种遍历的实现-3
|
2天前
|
存储
数据结构——二叉树四种遍历的实现-2
数据结构——二叉树四种遍历的实现
数据结构——二叉树四种遍历的实现-2
|
2天前
|
机器学习/深度学习
数据结构——二叉树四种遍历的实现-1
数据结构——二叉树四种遍历的实现
数据结构——二叉树四种遍历的实现-1
|
2天前
【数据结构】二叉树的三种遍历(非递归讲解)
【数据结构】二叉树的三种遍历(非递归讲解)
10 1
|
2天前
|
存储
【数据结构】二叉树相关oj题(一)
【数据结构】二叉树相关oj题(一)
10 1
|
2天前
|
存储 分布式数据库
[数据结构]~二叉树
[数据结构]~二叉树
|
2天前
|
C语言
【C语言/数据结构】二叉树(层序遍历|判断完全二叉树|性质)
【C语言/数据结构】二叉树(层序遍历|判断完全二叉树|性质)
280 52
|
2天前
【数据结构】二叉树-堆(top-k问题,堆排序,时间复杂度)
【数据结构】二叉树-堆(top-k问题,堆排序,时间复杂度)
17 4