数据结构c语言版第二版(严蔚敏)第五章笔记

news2025/1/9 1:58:22

目录

树和二叉树的定义

树的定义

树的基本术语

二叉树的定义

二叉树的性质和存储结构

二叉树的性质

二叉树的存储结构

顺序存储结构

链式存储结构

遍历二叉树和线索二叉树 

遍历二叉树

先序遍历

中序遍历

后序遍历

前序遍历的递归算法

中序遍历的递归算法

后序遍历的递归算法


 

树和二叉树的定义

树结构是一类重要的非线性数据结构。

树的定义

树是n(n >= 0)个节点的有限集,它或为空树(n == 0),或为非空树。

对于非空树:

1.有且仅有一个称之为根的节点

2.除根结点意外的其余节点可分为m(m > 0)个互不相交的有限集T1、T2、T3...Tn,其中每一个集合本身又是一棵树,并且称为根的子树

树的结构定义是一个递归的定义,即在树的定义中又用到树的定义。它道出了树的固有特性。树还有其他的表示方式,如:嵌套集合、广义表、凹入表示法

树的基本术语

1.节点:书中的一个独立的单元,包含一个数据元素及若干个指向其子树的分支

2.节点的度:节点拥有的子树数称为节点的度

3.树的度:树的度是树内各节点度的最大值

4.叶子:度为0的节点称为叶子或终端节点

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

6.双亲和节点:节点的子树的根称为该节点的孩子,相应地,该节点称为孩子的双亲

7.兄弟:同一个双亲的孩子之前互称兄弟

8.祖先:从根到该节点所经分支上的所有节点

9.子孙:以某节点为根的子树中的任意节点都称为该节点的子孙

10.层次:节点的层次从根开始定义,根为第一层,根的孩子为第二层。书中任意节点的层次等于其双亲结点的层次加一

11.堂兄弟:双亲在同一层的节点互为堂兄弟

12.树的深度:树中节点的最大层次称为树的深度或者高度

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

14.森林:m(m >= 0)棵互不相交的树的集合,对于树中每个节点而言,其子树的集合即为森林

就逻辑结构而言,任何一颗树都是一个二元组Tree = (root,F),其中root是数据元素,称作树的根节点;F是包含m(m >= 0)棵树的森林,F = (T1,T2,...Tn),其中Ti = (r1,Fi)称作根root的第i棵子树;当m != 0时,在树根和其子树森林之间存在下列关系

RF= \left \{ \left \langle root,r_{i} \right \rangle|i = 1,2...m;m > 0 \right \}

这个定义将有助于得到森林和树与二叉树之间转换的递归定义

二叉树的定义

二叉树是n(n >= 0)个节点所构成的集合,它或为空树(n == 0),或为非空树(n > 0)。对非空树T:

1.有且仅有一个称之为根的节点

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

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

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

2.二叉树的子树有左右之分,其次徐不能任意颠倒

二叉树的递归定义表示二叉树或为空,或由一个根节点加上两颗分别称为左子树和右子树的,互不相交的二叉树组成。由于这两棵树也是二叉树,则由二叉树的定义,它们也可以是空树,所以也就诞生了5中二叉树的基本形态:

空二叉树,仅有根节点的二叉树,右子树为空的二叉树,左子树为空的二叉树,左右节点均非空的二叉树

二叉树的性质和存储结构

二叉树的性质

二叉树具有以下重要性质

性质一:在二叉树的第i层上至多有2^{i - 1}(i >= 1)个节点

证明:利用归纳法容易证明此结论

i = 1时,只有一个根节点,显然是对的

现在假定所有的j(1 <= j < i)命题成立,即第j层上至多有2^{j - 1}个节点,那么可以证明j = i时也成立

由归纳假设,第i - 1层上至多有2^{i - 2}个节点,由于二叉树特性可得:每个节点的度至多为2,故在第i层上的最大节点数为在第i - 1层的最大节点数的二倍,也就是2*2^{i - 2} = 2^{i - 1}

性质二:深度为k的二叉树最多有2^{k} - 1(k >= 1)个节点

性质三:对任何一棵二叉树T,如果其终端节点数为n0,度为2的节点数为n2,则n0 = n2 + 1

下面介绍两种特殊形态的二叉树:满二叉树完全二叉树

满二叉树:深度为k且含有2^{k} - 1个节点的二叉树

满二叉树的特点是:每一层上的节点数都是最大节点数,即每一层i的节点数都具有最大值2^{i - 1}

可以对满二叉树的节点进行连续编号,约定编号从根节点起,自上而下,从左至右。由此可以引出完全二叉树的定义

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

完全二叉树的特点是:

1.叶子节点只可能在层次最大的两层上出现

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

性质四:具有n个节点的完全二叉树的深度为\left \lfloor log_{2}n \right \rfloor+ 1

\left \lfloor x \right \rfloor表示不大于x的最大整数,反之,也可表示为不小于x的最小整数

性质五:如果对一棵有n个节点的完全二叉树(其深度为\left \lfloor log_{2}n \right \rfloor+ 1)的节点按层序编号(从第1层到第\left \lfloor log_{2}n \right \rfloor+ 1层,每层从左到右),则对任意节点i(1 <= i <= n),以下结论成立

1.如果i = 1,则节点i是二叉树的根,无双亲;如果i > 1,则其双亲PARENT(i)是节点(i / 2)

2.如果2i > n,则节点i无左孩子(节点i为叶子节点),否则其左孩子LCHILD(i)是节点(2i)

3.如果2i + 1 > n,则节点无右孩子;否则其有孩子RCHILD(i)是节点(2i + 1)

二叉树的存储结构

顺序存储结构

#define MAXSIZE 100
typedef TELemType SqBiTree[MAXSIZE];
SqBiTree bt;

顺序存储结构使用一组地址连续的存储单元来存储数据元素,为了能够在存储结构中反映出节点之间的逻辑关系,必须将二叉树中的节 依照一定的规律安排在这组单元中

对于完全二叉树,只要从根起按层序存储即可,依次自上而下、自左至右存储节点元素,即将完全二叉树上编号为i的节点元素存储在如上定义的一维数组中下标为i - 1的分量中

对于一般二叉树,则应将其每个节点与完全二叉树上的节点相对照,存储在一维数组的相应分量中

由此可见,这种顺序存储结构仅适用于完全二叉树,因为在最坏的情况下,一个深度为k且只有k个节点的单支树(树中不存在度为2的节点)却需要长度为2^{k} - 1的一维数组,这造成了存储空间的极大浪费,因此对于一般二叉树,更适合采取链式存储结构

链式存储结构

设计不同的节点结构可构成不同形式的链式存储结构。由二叉树定义可知,二叉树的节点由一个数据元素和分别指向其左、右子树两个分支构成,则表示二叉树的链表中的节点至少包含三个域:数据域和左、右指针域。又是,为了便于找到节点的双亲,还可在节点结构中增加一个指向其双亲节点的指针域。利用这两种结构所得的二叉树的存储结构分别称为二叉链表和三叉链表。链表的头指针指向二叉树的根节点。容易证得,在含有n个节点的二叉链表中有n + 1个空链域

在不同的存储结构中,实现二叉树的操作方法也不同,例如找节点x的双亲PARENT(T,e),在三叉链表中很容易实现,而在二叉链表中则需从根指针出发巡查。由此,在具体应用中采用什么存储结构,除考虑二叉树的形态之外还应考虑需进行哪儿种操作。

typedef struct BiNode
{
	TElemTpye data;
	struct BiNode *lchild,*rchild;
}BiNode,*BiTree; 

遍历二叉树和线索二叉树 

遍历二叉树

遍历二叉树是指按某条搜索路径寻访树中的每个节点,使得每个节点被访问依次,而且仅被访问依次。访问的含义很广,可以是对节点进行各种处理,包括输出节点的信息,对节点进行运算和修改等。遍历二叉树是二叉树最基本的操作,也是二叉树其他各种操作的基础。遍历的是指是对二叉树进行线性化,即遍历的结果是讲非线性结构树中的节点排成一个线性序列。由于二叉树的每个节点都有可能有两颗子树,因此需要找寻一种规律,一遍是二叉树上的节点能排列在一个线性队列上,从而便于遍历

回顾二叉树的递归定义可知,二叉树由三个基本单元组成:根节点,左子树,右子树。因此,若能依次遍历这三部分,便是遍历了整个二叉树,假设用LDR分别表示左子树、访问根节点和右子树,那么可有LDR、LRD、DLR、DRL、RLD、RDL六中国遍历二叉树的方案,若限定先左后右,则只剩三种情况:DLR、LDR、LRD,我们称之为:先序遍历、中序遍历、后序遍历。基于二叉树的递归定义,可得下述遍历二叉树的递归算法定义:

  

先序遍历

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

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

1.访问根节点

2.先序遍历左子树

3.先序遍历右子树

先序遍历5.5二叉树得:-+a*b-cd/ef

中序遍历

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

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

1.中序遍历左子树

2.访问根节点

3.中序遍历右子树

中序遍历5.5二叉树得:a+b*c-d-e/f

后序遍历

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

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

1.后序遍历左子树

2.后序遍历右子树

3.访问根节点

后序遍历5.5二叉树得:abcd-*+ef/-

前序遍历的递归算法

void PreOrderTraverse(BiTree T)
{
	if(T)
	{
        cout << T -> data;
		PreOrderTraverse(T -> lchild);
		PreOrderTraverse(T -> rchild);
	}
} 

中序遍历的递归算法

void InOrderTraverse(BiTree T)
{
	if(T)
	{
		InOrderTraverse(T -> lchild);
		cout << T -> data;
		InOrderTraverse(T -> rchild);
	}
} 

后序遍历的递归算法

void PoOrderTraverse(BiTree T)
{
	if(T)
	{
		PoOrderTraverse(T -> lchild);
		PoOrderTraverse(T -> rchild);
        cout << T -> data;
	}
} 

也可利用我们之前学过的栈来将递归算法改为非递归算法

1.工作记录中包含两项,其一是递归调用的语句编号,其二是指向根节点的指针,则当栈顶记录中的指针非空时,应遍历左子树,即左子树根的指针进栈

2.若栈顶记录中的指针值为空,则应退至上一层,若是从左子树返回,则应当访问当前层(栈顶记录)中指针所指的根节点

3.若是从右子树返回,则表明当前层的遍历结束,应继续退栈。从另一个角度看,这意味着遍历右子树是不再需要保存当前层的根指针,直接修改栈顶记录中的指针即可

现给出中序遍历的非递归算法

void InOrderTreaverse(BiTree T)
{
	InitStack(S);
	BiNode p = T,q = new BiNode;
	while(p || !StackEmpty(S))
	{
		if(p)
		{
			Push(S,p);
			p = p -> lchild;
		}
		else
		{
			Pop(S,q);
			cout << q -> data;
			p = q -> rchild;
		}
	}
} 

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

二叉树的先序、中序、后续遍历是最常用的3种遍历方式。此外,还有一种按层次遍历二叉树的方式,这种方式按照“从上到下,从左到右”的顺序遍历二叉树,即先遍历二叉树第一层的节点,然后是第二层的节点,直至最底层的节点,对每一层的遍历按照从左到右的次序进行。层序遍历不是一个递归过程,层序遍历算法的实现可以借助队列这种数据结构 

 

 

 

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

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

相关文章

SARScape中用sentinel-1数据做SBAS-InSAR完整流程(2/2)

书接上回&#xff1a;SARScape中用sentinel-1数据做SBAS-InSAR完整流程&#xff08;1/2&#xff09; SARScape中用sentinel-1数据做SBAS-InSAR完整流程&#xff08;2/2&#xff09;7 反演第一步Inversion&#xff1a;First Step7.1 导入设置7.2 optional file7.3 parameters参数…

齐博x1用户登录接口

用户的登录主要涉及到小程序登录、APP的帐号密码登录、APP的微信开发平台帐号登录。 相应的地址是&#xff1a;http://qb.net/index.php/index/wxapp.login/index.html 涉及到的方法如下 上面的地址&#xff0c;默认是小程序的登录与注册。 http://qb.net/index.php/index/wxa…

matlab/simulink电力电子仿真傅里叶变换模块(fourier)测幅值相角的设置与使用

matlab/simulink电力电子仿真傅里叶变换模块&#xff08;fourier&#xff09;测幅值相角的设置与使用 今天要说的是一个可以测量信号的幅值和相角的模块&#xff0c;fourier&#xff0c;长下面这样&#xff1a; 有时候我们需要求某个信号的幅值或者相位&#xff0c;或求两个…

用文字描述给黑白照上色,这个免费网站火了!网友:比其他同类都好用

金磊 Alex 发自 凹非寺量子位 | 公众号 QbitAI这是清朝末代皇后婉容广为流传的一张老照片&#xff1a;如果让照片变成彩色的&#xff0c;会是什么样子&#xff1f;竟然没有什么违和感&#xff0c;百年前的老照片似乎在此刻变得鲜活了起来。而这张图上色的背后&#xff0c;并没有…

BUUCTF NewStarCTF 公开赛赛道Week5 Writeup

文章目录WEBGive me your photo PLZBabySSTI_ThreeUnsafe ApacheSo Baby RCE AgainFinal roundMISC最后的流量分析奇怪的PDF 2奇怪的文本Yesec no drumsticks 5qsdzs girlfriend 5WEB Give me your photo PLZ 可上传.htaccess AddType application/x-httpd-php .jpg然后上传…

干货!手把手教你穿透内网

干货&#xff01;手把手教你穿透内网干货&#xff01;手把手教你穿透内网cpolar内网穿透使用场景如何使用cpolar内网穿透&#xff1f; ↓↓1. 注册cpolar账号2. 安装cpolar内网穿透2.1 Windows系统2.2 Linux系统2.2.1 安装2.2.2 向系统添加服务2.2.3 启动服务2.2.4 查看服务状态…

生成二维码或条形码JavaScript脚本库

二维码或条形码在日常生活中现在应用已经非常普遍了&#xff0c;文章分享生成条形码和二维码的JavaScript库。 条形码 条形码是日常生活中比较常见的&#xff0c;主要用于商品。通俗的理解就是一串字符串的集合&#xff08;含字母、数字及其它ASCII字符的集合应用&#xff09…

【机器学习基础】 线性回归

线性回归1、线性回归定义2、线性回归题目示例3、推导公式4、误差5、似然函数6、线性回归评价指标7、梯度下降1、线性回归定义 经典统计学习技术中的线性回归和softmax回归可以视为 线性神经⽹络。给定训练数据特征 X 和对应的已知标签 y &#xff0c;线性回归的⽬标是找到⼀组权…

Seata安装启动

一、下载 https://github.com/seata/seata/releases/download/v1.4.2/seata-server-1.4.2.zip 二、启动 在安装路径下cmd seata-server.bat -h 127.0.0.1 -m file 三、作用 Seata是分布事务解决方案&#xff0c;seata保证微服务远程调用业务的原子性 Seata将为用户提供了 …

Spring Cloud LoadBalancer--负载均衡的原理(源码分析)

原文网址&#xff1a;Spring Cloud LoadBalancer--负载均衡的原理&#xff08;源码分析&#xff09;_IT利刃出鞘的博客-CSDN博客 简介 说明 本文介绍Spring Cloud LoadBalancer负载均衡的原理。 SpringCloud从2020版本开始移除了对Ribbon的依赖&#xff0c;官方使用Spring Cl…

VsCode中一些可以让工作“事半功倍”的插件

1.GitLens — Git supercharged 这个插件可以查看代码修改的消息&#xff0c;比如是谁修改的以及修改时间 2.Chinese (Simplified) (简体中文) 简体中文&#xff0c;这个可以说是装的最多的一款插件了 3.Auto Close Tag 标签自动补全 4.Auto Rename Tag&#xff1a;自动完…

2021第7届中国大学生程序设计竞赛CCPC广州站, 签到题4题

文章目录I.Pudding StoreH.Three IntegersC.NecklaceF.Cactus补题链接&#xff1a;https://codeforces.com/gym/103415 I.Pudding Store I. Pudding Store time limit per test2.0 s memory limit per test512 megabytes inputstandard input outputstandard output 159 is a…

快速排序图解(两种思想)

七大排序之快速排序 文章目录七大排序之快速排序前言一、《算法导论》中的分区思想1.1 算法思想1.2 代码实现二、Hoare挖坑法2.1 算法思想2.2 代码实现三、算法分析四、注意事项总结前言 博主个人社区&#xff1a;开发与算法学习社区 博主个人主页&#xff1a;Killing Vibe的博…

【每天学习一点新知识】网络安全--截获攻击

截获攻击原理和后果 原理 若正常传输路径为终端A到终端B&#xff0c;黑客首先改变传输路径为终端A—黑客终端—终端B&#xff0c;使得传输信息必须经过黑客终端&#xff0c;黑客终端就可以截获终端A传输给终端B的消息。 后果 目前很多访问过程采用明码方式传输登录的用户名和密…

C++入门基础(下)

目录 引用 引用概念 引用特性 1.引用在定义时必须初始化 2.一个变量可以有多个引用 3.引用一旦引用一个实体&#xff0c;再不能引用其他实体. 常引用 使用场景 1.作为参数使用 2.作为返回值使用 引用和指针的区别 内联函数 内联函数的概念 内联函数特性 宏的优缺点 auto关键字 …

scala spark dataframe 时间加减

参考Adding 12 hours to datetime column in Spark 只针对标准化时间戳 yyyy-MM-dd HH:mm:ss 如果是 yyyy-MM-dd HH:mm 转换后会自动补到 HH:mm:ss ss位补0 时间英文简写查询 HOUR 代表小时 MINUTE 代表分钟 SECOND 代表秒 DAY MONTH YEAR 正数代表向后 负数代表向前 …

AI绘画突然爆火?快速体验二次元画师NovelAI(diffusion)

目录0 写在前面1 diffusion vs GAN2 NovelAI3 AI绘画环境搭建4 体验AI创作0 写在前面 机器学习强基计划聚焦深度和广度&#xff0c;加深对机器学习模型的理解与应用。“深”在详细推导算法模型背后的数学原理&#xff1b;“广”在分析多个机器学习模型&#xff1a;决策树、支持…

到了30岁,我才有了深刻的感悟:千万不要一辈子靠技术生存

千万不要一辈子靠技术生存&#xff0c;这句话&#xff0c;我也是到了快30岁才有了深刻认知。 当我20多岁&#xff0c;年收入2-3W的时候&#xff0c;我会认为说这话的人都是自身技术不咋地&#xff0c;想靠技术吃饭吃不了。 然而&#xff0c;快30岁了&#xff0c;虽然技术小有…

【Java】之IO流

个人主页&#xff1a;天寒雨落的博客_CSDN博客-C,CSDN竞赛,python领域博主 特别标注&#xff1a;仅为自己的学习记录笔记&#xff0c;方便复习和加深记忆&#xff0c;仅供借鉴参考&#xff01; 目录 IO流概述 IO流分类 按数据的流向 按数据类型 字符流 字节流 字节流写数…

【Linux】虚拟机安装Ubuntu后的一些通用设置

文章目录前言一、虚拟机缩放设置二、实现本机和虚拟机之间复制粘贴共享三、ubuntu中vi文件时方向键等问题四、虚拟机扩容五、时区和时间格式设置六、防火墙相关七、中文输入法问题八、虚拟机和主机之间的互通前言 主要是记录虚拟机中安装ubuntu后一些常规设置操作。 一、虚拟…