数据结构——树和二叉树最全总结(期末复习必备)

news2025/1/15 6:48:07

目录

树和二叉树

 树的基本术语(均以上图b为例):

遍历二叉树:

线索二叉树: 

 树的存储结构:

树与二叉树的转换(利用的就是把二叉树和树表示成相同的二叉链表):

森林与二叉树的转换:

哈夫曼树


树和二叉树

树(Tree)是n(n≥0)个结点的有限集,它或为空树(n=0);或为非空树,对于非空树T:

(1)有且仅有一个称之为根的结点;

(2)除根结点以外的其余结点可分为m(m>0)个互不相交的有限集T_{1},T2……,Tm,其中每个集合本身又是一棵树,并且称为根的子树(SubTree )。

总:树的定义是一个递归定义,即在树的定义中又用到树的定义,它道出了树的固有特性。

 树的基本术语(均以上图b为例):

(1)结点:树中的一个独立单元。包含一个数据元素及若干指向其子树的分支,如上图(b)中的A、B、C、D等。

(2)结点的度:结点拥有的子树数称为结点的度。例如,A的度为3,C的度为1,F的度为0。

 (3)树的度:树的度是树内各结点度的最大值。图(b)所示的树的度为3.

(4)叶子:度为0的结点称为叶子或终端结点。结点K、L、F、G、M、I、J都是树的叶子。

(5)非终端结点:度不为0的结点称为非终端结点或分支结点。除根结点之外,非终端结点也称为内部结点。

(6)双亲和孩子:结点的子树的根称为该结点的孩子,相应地,该结点称为孩子的双亲。例如,B的双亲为A,B的孩子有E和F。

( 7)兄弟:同一个双亲的孩子之间互称兄弟。例如,H、I和J互为兄弟。

(8)祖先:从根到该结点所经分支上的所有结点。例如,M的祖先为A、D和H。

(9)子孙:以某结点为根的子树中的任一结点都称为该结点的子孙。如B的子孙为E、K、L和F。

(10))层次:结点的层次从根开始定义起,根为第一层,根的孩子为第二层。树中任一结点的层次等于其双亲结点的层次加1。

( 11)堂兄弟:双亲在同一层的结点互为堂兄弟。例如,结点G与E、F、H、I、J互为堂兄弟。

(12)树的深度:树中结点的最大层次称为树的深度或高度。图(b)所示的树的深度为4。

(13)有序树和无序树:如果将树中结点的各子树看成从左至右是有次序的(即不能互换),则称该树为有序树,否则称为无序树。在有序树中最左边的子树的根称为第一个孩子,最右边的称为最后一个孩子。

(14)森林:是m(m≥0)棵互不相交的树的集合。对树中每个结点而言,其子树的集合即为森林。由此,也可以用森林和树相互递归的定义来描述树。

二叉树(Binary Tree)是n(n\geqslant0)个结点所构成的集合,它或为空树(n=0);或为非空树,对于非空树T:

(1)有且仅有一个称之为根的结点

(2)除根结点以外的其余结点分为两个互不相交的子集T1和T2,分别称为T的左子树和右子树,且T1和T2本身又都是二叉树。

二叉树与树一样具有递归性质,二叉树与树的区别主要有以下两点:

(1)二叉树每个结点至多只有两棵子树(即二叉树中不存在度大于2的结点);

(2)二叉树的子树有左右之分,其次序不能任意颠倒。

二叉树的五种和那个基本形态:

二叉树的性质:

  1.  在二叉树的第i层上至多有2^{i-1}个结点
  2. 深度为k的二叉树至多有2^{k}-1个结点
  3. 对于任何一颗二叉树,如果其终端节点数为n,度为2的节点数为m,则n=m+1

满二叉树:深度为k且含有2^{k}-1个结点的二叉树,即每i层都有2^{i-1}个结点

完全二叉树:深度为k的,有n个结点的二叉树,当且仅当其每一个结点都与深度为k的满二叉树中编号从1至n的结点一一对应时,称之为完全二叉树。

完全二叉树的特点是:

(1)叶子结点只可能在层次最大的两层上出现;

(2)对任一结点,若其右分支下的子孙的最大层次为l,则其左分支下的子孙的最大层次必为1或l+1。

二叉树的存储结构:

  • 顺序存储结构

顺序存储结构使用一组地址连续的存储单元来存储数据元素,对于完全二叉树只要从根起按层序存储即可,依次自上而下,自左至右存储节点元素。由此可见,顺序存储结构仅适用于完全二叉树;存储普通的二叉树遇多个无元素的结点会导致存储空间的极大浪费。对于一般二叉树更适合采取下面的链式存储结构

  • 链式存储结构

对于二叉树中的结点包括一个数据域和三个指针域(左子树和右子树和父母)

遍历二叉树:

遍历二叉树是按照某条搜索路径访问树中的每一个节点(仅访问一次),想要对二叉树进行操作首先得遍历。二叉树是由三个基本单元组成:根节点、左子树、右子树;则遍历二叉树的思路就是依次遍历左根右等6种不同的顺序,在此介绍三种:

先序遍历二叉树的操作定义如下:

若二叉树为空,则空操作;否则

(1)访问根结点;

(2)先序遍历左子树;

(3)先序遍历右子树。

中序遍历二叉树的操作定义如下:

若二叉树为空,则空操作;否则

(1)中序遍历左子树;

(2)访问根节点

(3)中序遍历右子树

后序遍历二叉树的操作定义如下:

若二叉树为空,则空操作;否则

(1)后序遍历左子树

(2)后序遍历右子树

(3)访问根节点

中序遍历递归算法:

 利用栈将递归算法写成非递归算法:

 无论是递归还是非递归遍历二叉树,因为每个节点被访问一次,则不论按照哪一种次序进行遍历,对含有n个结点的二叉树,其时间复杂度均为O(n);所需辅助空间为遍历过程中栈的最大容量,最坏情况下为n,则空间复杂度为O(n)

给定一个二叉树和遍历规则我们可以唯一确定的写出序列;相反由二叉树的先序序列(或后序序列)和中序序列均能唯一确定一棵二叉树。

复制二叉树:

统计二叉树中结点的个数:

线索二叉树: 

 在上述二叉树中每个节点包含左右子树信息。但无法知道结点在任意序列中的前驱和后继,这种信息只有在遍历二叉树后得到,线索二叉树就是来解决该问题;由于有n个结点的二叉链表中必定有n+1个空链域,因此我们可以充分利用空链域来存放结点的前驱和后继信息。防止混淆,我们在二叉树的结点增加两个标志域。

若标志域是0表示的是指向该结点的左(右)孩子,标志域是1表示的是前驱(后继)

二叉树的二叉线索类型定义如下:

 

 以这种结点结构构成的二叉链表作为二叉树的存储结构,叫做线索链表,其中指向结点前驱和后继的指针叫做线索;加上线索的二叉树称之为线索二叉树

线索二叉树的实质就是把二叉链表中的空指针改为指向前驱或者后继的线索,而前驱或后继的信息只有在遍历时才能得到。

以结点p为根的子树中序线索化

 带头结点的二叉树中序线索化

 遍历中序线索二叉树:

 树的存储结构:

  • 双亲表示法

该方法以一组连续的存储单元存储树的结点,每个结点除了数据域data外,还附有一个parent域用以指示双亲结点在数组中的位置。

C语言类型描述:

 优点:找双亲方便

缺点:找孩子不方便

  • 孩子表示法

把每个结点的孩子结点排列起来,看成是一个线性表,且以单链表做存储结构,则n个结点有n个孩子链表(叶子的孩子链表为空表);而n个头指针又组成一个线性表,为了便于查找,可采用顺序存储结构。

typedef struct CTNode{
    int child;
    struct CTNode *next;}*ChildPtr;
typedef struct{
    TElemType data;
    ChildPtr firstchild;}CTBox;
typedef struct{
    CTBox nodes[MAX_TREE_SIZE];
    int n,r;}CTree;

找孩子方便,找父母麻烦

  • 孩子兄弟表示法

又称二叉树表示法,或二叉链表表示法,即以二叉链表做树的存储结构;链表中结点的两个链域分别指向该节点的第一个孩子结点和该结点的下一个兄弟结点,分别命名为firstchild域和nextsibling域

树与二叉树的转换(利用的就是把二叉树和树表示成相同的二叉链表):

树转换成二叉树的思路(口诀:兄弟相连留长子):

  1. 加线:在兄弟之间加一连线
  2. 抹线:对每个结点,除了其左孩子外,去除其与其孩子之间的关系
  3. 旋转:以树的根节点为轴心,将整树顺时针转45^{\circ}

给定一棵树,可以找到 唯一的一棵二叉树与之对应

二叉树转换成树思路:

  1. 加线:若p结点是双亲结点的左孩子,则将p的右孩子、右孩子的右孩子……沿分支找到的所有右孩子,都与p的双亲用线连接起来
  2. 抹线:抹掉原二叉树中双亲与右孩子之间的连线
  3. 调整:将结点按层次排列,形成树结构

森林与二叉树的转换:

森林转换成二叉树思路:(口诀:树变二叉根相连)

  1. 将各棵树分别转换成二叉树
  2. 将每棵树的根结点用线相连
  3. 以第一棵树根节点为二叉树的根,再以根节点为轴心,顺时针旋转,构成二叉树型结构

二叉树转换成森林思路:(口诀:去掉全部右孩线,孤立二叉再还原)

  1. 抹线:将二叉树中根节点与其右孩子连线,及沿右分支搜索到的所有右孩子间连线全部抹掉,使之变成孤立的二叉树
  2. 还原:将孤立的二叉树huan'yuan

树和森林的遍历:

树的遍历:

  • 先根(次序)遍历:若树不空,则先访问根节点,然后依次先根遍历各棵子树
  • 后根(次序)遍历:若树不空,则先依次后根遍历各棵子树,然后访问根节点
  • 按层次遍历:若树不空,则自上而下自左至右访问树中每个结点

将森林看作三部分构成:

  1. 森林中第一棵树的根节点
  2. 森林中第一棵树的子树森林
  3. 森林中其它树构成的森林

森林的遍历:

  • 先序遍历:若森林不空,则访问森林中第一棵树的根节点;然后先序遍历森林中第一棵树的子树森林;最后先序遍历森林中(除第一棵树之外)其余树构成的森林(即从左至右对森林中的每一棵树进行先根遍历)
  • 中序遍历:若森林不空,则中序遍历森林中第一棵树的子树森林;然后访问森林中第一棵树的根节点;最后中序遍历森林中(除第一棵树之外)其余树构成的森林 (依次从左至右对森林中的每一棵树进行后根遍历)

哈夫曼树

路径:从树中一个结点到另一个结点之间的分支构成这两个结点之间的路径。

路径长度:路径上的分支数目称作路径长度。

树的路径长度:从树根到每一结点的路径长度之和。

权:将树中结点赋给一个有着某种含义的数值,则称这个数值称为该结点的权

结点的带权路径长度:从该结点到树根之间的路径长度与节点上权的乘积。

树的带权路径长度:树中所有叶子结点的带权路径长度之和

哈夫曼树:又称作最优树;是带权路径长度最短的树(“带权路径长度最短”是在“度相同”的树中比较而得到的结果,因此有最优二叉树、最优三叉树等等)

提醒:满二叉树不一定是哈夫曼树;在节点数目相同的二叉树中,完全二叉树是路径长度最短的二叉树;注意区分树的路径长度和树的带权路径长度是完全不一样的

哈夫曼树的构造方法:

1.根据n个给定的权值{W1,W2,W3……Wn}构成n棵二叉树的森林,其中森林中的每一棵树都是一个根节点

2.在森林中选取两棵根结点的权值最小的成为左右子树,构造一棵新的二叉树,且设置新的二叉树的根节点的权值为其左右子树上根结点的权值之和。

3.在F中删除这两棵树,同时将新得到的二叉树加入森林中

4.重复第2步和第3步,直到森林中只有一棵树为止,这棵树即为哈夫曼树

口诀:构造森林全是根;选用两小造新树;删除两小添新人;重复2、3剩单根

总结:初始时有n棵二叉树,要经过n-1次合并最终形成哈夫曼树;经过n-1次合并产生n-1个新结点,且这n-1个新结点都是具有两个孩子的分支结点;所以,哈夫曼树共有2n-1个结点,因为每次合并都会产生一个新的结点;且其所有的分支节点的度均不为1

哈夫曼树构造算法的实现

  • 顺序存储结构(一维结构数组)

哈夫曼树的存储表示:

算法实现:

 

哈夫曼编码思路:

  1. 统计字符集中每个字符在电文中出现的平均概率(概率越大,要求编码越短)
  2. 利用哈夫曼树的特点:权重越大的叶子离根越近;将每个字符的概率值作为权值,构造哈夫曼树。则概率越大的结点,路径越短。
  3. 在哈夫曼树的每个分支上标上0或1:结点的左分支标0,右分支标1,把从根到每个叶子的路径上的标号连接起来,作为该叶子代表的字符的编码

性质:

  • 哈夫曼编码是前缀码(前缀码就是该编码在解码时不会产生歧义)
  • 哈夫曼编码是最优前缀码

哈夫曼编码的算法实现:

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/74585.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

KubeSphere 接入外部 Elasticsearch 最佳实践

作者:张坚,科大讯飞开发工程师,云原生爱好者。 大家好,我是张坚。今天来聊聊如何在 KubeSphere 中集成外置的 ES 组件。 KubeSphere 在安装完成时候可以启用日志组件,这样会安装 ES 组件并可以收集所有部署组件的日志…

索引优化学习

背景 最近做查询优化,学到的。字段长度,索引长度联合索引计算是否使用范围查询使用索引 字段长度(varchar) 只谈论varchar:首先我们建表varchar(20) 中的20是字符数。看你的数据库编码 执行:show creat…

支持多种网关类型!米尔基于Zynq-7010/20开发平台工业网关设计应用

随着工业物联网的飞速的发展,5G时代的到来,工业控制系统在生产领域应用越来越广泛,工业物联网为未来工业控制系统灵活性和可扩展性的需求提供了支持。工业物联网使我们的生产数据可以进行规模化集中存储,并利用高速采集、云计算等…

ChatGPT国产平替出现了:APP商店就能下载,还可给AI加人设,背后公司刚成立3个月...

明敏 发自 凹非寺量子位 | 公众号 QbitAIChatGPT太火爆谁不想上手试试?但注册复杂、服务器拥挤……着实有点麻烦。不过很快就有极客网友指路,说国内其实已经有类似的APP上线了,也是上知天文下知地理的那种。比如聊聊《三体》,还会…

Transformer 训练优化

前言 自 BERT 出现以来,NLP 领域已经进入了大模型的时代,大模型虽然效果好,但是毕竟不是人人都有着丰富的 GPU 资源,在训练时往往就捉襟见肘,出现显存 out of memory 的问题,或者训练时间非常非常的久&…

web期末大作业:基于html+css+js制作深圳大学网站(13页) 学校班级网页制作模板 学生静态HTML网页源码

🎉精彩专栏推荐 💭文末获取联系 ✍️ 作者简介: 一个热爱把逻辑思维转变为代码的技术博主 💂 作者主页: 【主页——🚀获取更多优质源码】 🎓 web前端期末大作业: 【📚毕设项目精品实战案例 (10…

【easypoi 模板导出嵌套 list 问题】

easypoi 模板导出多 list 问题1背景1.1 espoi 模板定义1.2 导出结果1.3发现共享单车的数据没有显示,手动操作取消共享单车的单元格合并1.4手动取消单元格合并后数据,正常再把框线画好1.5 代码操作:用 esaypoi 处理到1.2,用基础 po…

聊聊与前端工程师天然互补的 Serverless

作为前端工程师,我们的使命是为用户提供良好的前端用户体验。随着云原生时代的到来,显而易见的,我们能做的更多了。Serverless 产品的特点是免运维、按量付费和自适应弹性,所以我们可以利用云上的各种 Serverless 能力&#xff0c…

《程序员的自我修养》程序实现的两大环境

学习内容 翻译环境 预处理 编译 汇编 链接 执行环境 在标准C的任何一种实现中,都存在两个不同的环境: 1.翻译环境:在这个环境中,源代码被翻译成为可执行的机器指令。 2.执行环境:用于执行实际的代码 在VS2022中&…

字符串的读入方式

文章目录1、scanf2、fgets()3、cin4、cin.getline()5、getline()1、scanf scanf只能读入不带空格的字符串,遇到空格则结束。scanf只能读入字符数组,不能读入string。scanf在读入的时候,会自动在字符串的末尾加上’\0’。定义字符数组长度时&…

虚表指针在C++类的继承/多态与重载中的基本逻辑

文章目录前言重载继承虚函数虚函数表动态绑定的实现析构函数构造函数多态子类直接继承父类的方法,不覆盖多重继承纯虚函数前言 C在C语言的基础上增加了类的概念,而类的最关键的特性就是三个: 继承多态重载 这篇文章想接着上两篇C相关的文章…

[附源码]Python计算机毕业设计SSM基于推荐算法的汽车租赁系统(程序+LW)

项目运行 环境配置: Jdk1.8 Tomcat7.0 Mysql HBuilderX(Webstorm也行) Eclispe(IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持)。 项目技术: SSM mybatis Maven Vue 等等组成,B/S模式 M…

2023最新SSM计算机毕业设计选题大全(附源码+LW)之java动物爱心救助平台s9dza

首先选择计算机题目的时候先看定什么主题,一般的话都选择当年最热门的话题进行组题,就比如说,今年的热门话题有奥运会,全运会,残运会,或者疫情相关的,这些都是热门话题,所以你就可以…

HTML做一个抗疫逆行者感动人物页面(纯html代码)

🎉精彩专栏推荐 💭文末获取联系 ✍️ 作者简介: 一个热爱把逻辑思维转变为代码的技术博主 💂 作者主页: 【主页——🚀获取更多优质源码】 🎓 web前端期末大作业: 【📚毕设项目精品实战案例 (10…

AI公平性研究进展(2022)

最近AI公平性方面出了不少新的研究成果,如有遗漏,欢迎补充↓↓↓↓ 公平性提升 MAAT: A Novel Ensemble Approach to Addressing Fairness and Performance Bugs for Machine Learning Software, FSE, 2022. 利用模型集成的方式获得公平的预测结果&am…

Windows下如何查看某个端口被谁占用

开发时经常遇到端口被占用的情况,这个时候我们就需要找出被占用端口的程序,然后结束它,本文为大家介绍如何查找被占用的端口。 1、打开命令窗口(以管理员身份运行) 开始—->运行—->cmd,或者是 windowR 组合键&#xff0c…

含冰蓄冷空调的冷热电联供型微网多时间尺度优化调度(Matlab代码实现)

💥💥💞💞欢迎来到本博客❤️❤️💥💥 🏆博主优势:🌞🌞🌞博客内容尽量做到思维缜密,逻辑清晰,为了方便读者。 ⛳️座右铭&a…

开关电源 - LLC电路的谐振工作模态浅析

LLC电路的谐振工作模态浅析 LLC谐振电路是常用的拓扑,广泛应用在目前的热门应用中,本文主要从典型谐振状态来分析其基本工作过程,后续我们将逐步扩展到其它工作状态。 一、电路工作基本条件及电路组成 图1 电路主要组成部分 如上图1所示&a…

c#入门-运算符

可用运算符 值的运算也是自定义的,但是这个就看不出了。 要么翻源文件,要么翻说明书才知道一个类型能怎么运算。 但是一个默认情况下的类型,是没有定义任何运算符的。 也就是说你适用的运算一定是由人先写出来的,没有系统默认的…

MYSQL的索引事务

文章目录1:索引1.1:概念:1.2 作用:1.3 使用场景:1.4 使用:补充:**2:了解一下B树:****3:了解一下B树**4:为什么使用B树/B树来实现索引2&#xff1a…