二叉树03(数据结构初阶)
文章目录
- 一:实现链式结构二叉树
- 1.1前中后序遍历
- 1.1.1遍历规则
- 1.1.2代码实现
- 1.2结点个数以及高度等
- 1.2.1二叉树结点个数
- 1.2.2二叉树叶子结点个数
- 1.2.3二叉树第k层结点个数
- 1.2.4二叉树的深度/高度
- 1.2.5 二叉树查找值为x的结点
- 1.2.6二叉树的销毁
- 1.3层序遍历
- 1.4判断是否为完全二叉树
- 二:结语
欢迎大家阅读我的博客,给生活加点impetus,!!
今天我们学习二叉树的实现,感受递归的魅力!!
一:实现链式结构二叉树
⽤链表来表⽰⼀棵⼆叉树,即⽤链来指⽰元素的逻辑关系。通常的⽅法是链表中每个结点由三个域组成,数据域和左右指针域,左右指针分别⽤来给出该结点左孩⼦和右孩⼦所在的链结点的存储地址。
我们先来定义链式结构二叉树的结构。
链式结构二叉树是由结点组成的,定义二叉树的结构就是定义结点的结构。
在数据结构初阶的学习过程中中,前期对于二叉树的插入代码不会深入研究,在后期c++阶段学习红黑树,平衡树等会对二叉树数据的插入操作进行研究。
我们需要了解二叉树的遍历方式有哪些
1.1前中后序遍历
为了后方我们能够理解深刻,我们先依据下列图示来手动创建一个二叉树
这里我们的存储类型为char类型,因为存储的是字符。
创建结点:
构造链式二叉树:
接下来我们来深入讲解各种遍历方法的规则。
1.1.1遍历规则
按照规则,⼆叉树的遍历有:前序/中序/后序的递归结构遍历:
1)前序遍历(Preorder Traversal 亦称先序遍历):访问根结点的操作发⽣在遍历其左右⼦树之前
访问顺序为:根结点、左⼦树、右⼦树
2)中序遍历(Inorder Traversal):访问根结点的操作发⽣在遍历其左右⼦树之中(间)
访问顺序为:左⼦树、根结点、右⼦树
3)后序遍历(Postorder Traversal):访问根结点的操作发⽣在遍历其左右⼦树之后
访问顺序为:左⼦树、右⼦树、根结点
接下来依靠规则自行先思考上述二叉树的遍历顺序,先来说明结果
前序遍历:首先从根结点进入,依照前序遍历规则–根左右首先A结点是根结点,可以直接打印,左方为B结点,B同样也是根结点,所以打印B后会继续向下遍历,直到找到不为根结点的结点,其实就是叶子结点NULL,右的话就是D结点右子树NULL,此时B结点的整个左子树遍历完全,同样方法遍历右子树,直至A结点的左子树全部遍历完毕。同理遍历右子树。
顺序为A B D NULL NULL NULL C E NULL NULL F NULL NULL
中序遍历:首先从根结点进入,依照前序遍历规则–左根右首先A结点是根结点,不能打印,会继续向左遍历,直至找到不为任何子树的结点(度为0的结点,即叶子结点),先打印D的左子树,再打印D,最后打印D的右子树,随后B的整个左子树打印完毕,打印B后,再打印B的右子树,随后A的整个左子树打印完毕打印A后,随后打印A的右子树,同理。
顺序为NULL D NULL B NULL A NULL E NULL C NULL F NULL
后序遍历:首先从根结点进入,依照前序遍历规则–左右根首先A结点是根结点,不能打印,会继续向左遍历,直至找到不为任何子树的结点(度为0的结点,即叶子结点),先打印D的左子树,再打印D的右子树,再打印D,随后B的整个左子树打印完毕,打印B的右子树NULL,随后打印B,A的整个左子树打印完毕,同理打印A的右子树。
打印顺序:NULL NULL D NULL B NULL NULL E NULL NULL F C A
所以遍历时我们需要注意:
1:除叶子结点以外,其他结点都是根结点,当层次很多时,1到h-1层都是层次>=2的树或子树。
2:遍历时主要是涉及递归函数
1.1.2代码实现
前序遍历:
中序遍历:
后序遍历:
图解(前序遍历):
细节:
1:遍历只能从根结点开始遍历,且一定要根据口诀的顺序
2:遍历都是先经过包含叶子结点的二叉树
3:叶子结点其实是递归的结束条件
4:递推回归时与函数栈帧创建的联系
5:遍历时某结点看做根结点,该棵树一定要延伸到叶子结点,才能看做一棵树。
1.2结点个数以及高度等
1.2.1二叉树结点个数
大部分人想到的都是设置计数器,那我们来看看这个算法的好坏。
方法一:
所以这个地方我们需要传地址,而且,size既不能做全局变量,也不能做局部变量。那我们索性添加一个参数。
来看下面这段代码:
方法二:直接来返回int
思路:结点个数=根结点(1)+左子树结点+右子树结点
我们来画图演示一下:
逐步递推,逐步回归、
方法二同样能够实现预期结果,并且思路更加简单。
总结:
1:函数栈帧销毁:函数执行完全或者return提前返回
:2:局部变量+static生命周期延长,仍然存在作用域,上例修饰size时仍然错误
1.2.2二叉树叶子结点个数
思路**:左子树叶子结点个数+右子树叶子结点个数**
1.2.3二叉树第k层结点个数
思路:深度优先遍历,k自减,准确找到第k层,最后返回个数
1.2.4二叉树的深度/高度
思路:根结点+max{左子树,右子树}
最好使用变量接收一下,因为我们需要比较大小
1.2.5 二叉树查找值为x的结点
思路:查找左子树与右子树,同时该结点是否为X
1.2.6二叉树的销毁
思路:只能后序遍历,因为每一个结点malloc都要释放,如果先释放根结点,孩子结点就找不到了。
注意:优先级高低:->大于*。
free之后及时置空,防止成为野指针。
1.3层序遍历
前方的所有遍历都是涉及前中后序遍历,都是深度优先遍历。
下面我们来讲解广度优先遍历(层序遍历),我们来看这样一个题目:
如果我们仍然按照深度优先来遍历的话,肯定是不行的,因为只有当该部分函数栈帧销毁之后,才会继续进入到上一个函数栈帧。
思路:借助数据结构–队列,根结点入队列,循环判断队列是否为空
不为空取队头出队头,将队头的左右且不为空的孩子入队列。
我们来画图演示一下:
再来看一张:
下面是代码部分:
细节
1:我们涉及到了Queue,我们需要重新添加Queue.c和Queue.h文件。
2:队列中存储的是二叉树结点指针类型,需要将Queue中定义的int改为struct BTNode。
3:typedef重定义的是数据类型,一定要加struct加以声明,只写BTNode会报错。
4:存储指针是因为结构体访问成员是指针类型(左孩子,右孩子),如果直接存储结构体会报错。
5:如果也把空孩子加入,就会退出循环。
1.4判断是否为完全二叉树
性质:
1:除最后一层外,其余结点的度都为2
2:有序
我们需要使用层序遍历
思路:层序遍历,根结点入队列,循环判断队列是否为空不为空取队头出队头,将队头的左右孩子入队列,不管是不是空遇到空停止循环,比较队列剩余元素,如果还有非空元素,则为非完全二叉树,反之是完全二叉树。
我们来画图理解观察区别,层序遍历部分我将不再演示:
接下来来看代码:
细节:
1:层序遍历入得都是top(队头)的左右孩子,如果写成root->left,这将面临死循环。
:2:这里的队列并不是依次相连的,中间有很多断开的,而最开始的Queue结点是一个一个依次相连的
二:结语
最后,感谢大家的阅读,欢迎大家有更多的思路与我交流,同时不足之处欢迎指正!!
逆水行舟–不进则退!!!加油,希望早日看见最好的自己!!
相关文章:

二叉树03(数据结构初阶)
文章目录 一:实现链式结构二叉树1.1前中后序遍历1.1.1遍历规则1.1.2代码实现 1.2结点个数以及高度等1.2.1二叉树结点个数1.2.2二叉树叶子结点个数1.2.3二叉树第k层结点个数1.2.4二叉树的深度/高度1.2.5 二叉树查找值为x的结点1.2.6二叉树的销毁 1.3层序遍历1.4判断是…...

ComfyUI工作流 图像反推生成人像手办人像参考(SDXL版)
文章目录 图像反推生成人像手办人像参考SD模型Node节点工作流程效果展示开发与应用图像反推生成人像手办人像参考 本工作流旨在通过利用 Stable Diffusion XL(SDXL)模型和相关辅助节点,实现高效的人像参考生成和手办设计。用户可通过加载定制的模型、LORA 调整和控制节点对…...

【01】共识机制
BTF共识 拜占庭将军问题 拜占庭将军问题是一个共识问题 起源 Leslie Lamport在论文《The Byzantine Generals Problem》提出拜占庭将军问题。 核心描述 军中可能有叛徒,却要保证进攻一致,由此引申到计算领域,发展成了一种容错理论。随着…...
sentinel的限流原理
Sentinel 的限流原理基于 流量统计 和 流量控制策略,通过动态规则对系统资源进行保护。其核心设计包括以下几个关键点: 流量统计模型:滑动时间窗口 Sentinel 使用 滑动时间窗口算法 统计单位时间内的请求量,相比传统的固定时间窗…...

ZOJ 1007 Numerical Summation of a Series
原题目链接 生成该系列值的表格 对于x 的 2001 个值,x 0.000、0.001、0.002、…、2.000。表中的所有条目的绝对误差必须小于 0.5e-12(精度为 12 位)。此问题基于 Hamming (1962) 的一个问题,当时的大型机按今天的微型计算机标准来…...
『 C 』 `##` 在 C 语言宏定义中的作用解析
文章目录 ## 运算符的基本概念可变参数宏与 ## 的应用可变参数宏简介## 处理可变参数的两种情况可变参数列表为空可变参数列表不为空 示例代码验证 在 C 和 C 编程里,宏定义是个很有用的工具。今天咱们就来聊聊 ## 这个预处理器连接运算符在宏定义中的作用ÿ…...
独立成分分析 (ICA):用于信号分离或降维
人工智能例子汇总:AI常见的算法和例子-CSDN博客 独立成分分析 (Independent Component Analysis, ICA) 是一种用于信号分离和降维的统计方法,常用于盲源分离 (Blind Source Separation, BSS) 问题,例如音频信号分离或脑电信号 (EEG) 处理。…...
为什么会有函数调用参数带标签的写法?Swift函数调用的参数传递需要加前缀是否是冗余?函数调用?函数参数?
为什么会有函数调用参数带标签的写法? ObjC函数参数形式与众不同,实参前会加前缀,尤其参数很多的情况,可读性很强。例如: [person setAge: 29 setSex:1 setClass: 35]; 这种参数前面加前缀描述也被叫标签(Label). 注意࿰…...

实际操作 检测缺陷刀片
号he 找到目标图像的缺陷位置,首先思路为对图像进行预处理,灰度-二值化-针对图像进行轮廓分析 //定义结构元素 Mat se getStructuringElement(MORPH_RECT, Size(3, 3), Point(-1, -1)); morphologyEx(thre, tc, MORPH_OPEN, se, Point(-1, -1), 1); …...

使用Pygame制作“青蛙过河”游戏
本篇博客将演示如何使用 Python Pygame 从零开始编写一款 Frogger 风格的小游戏。Frogger 是一款早期街机经典,玩家需要帮助青蛙穿越车水马龙的马路到达对岸。本示例提供了一个精简原型,包含角色移动、汽车生成与移动、碰撞检测、胜利条件等关键点。希望…...

BUU17 [RoarCTF 2019]Easy Calc1
自用 源代码 $(#calc).submit(function(){$.ajax({url:"calc.php?num"encodeURIComponent($("#content").val()),type:GET,success:function(data){$("#result").html(<div class"alert alert-success"><strong>答案:&l…...

堆的实现——对的应用(堆排序)
文章目录 1.堆的实现2.堆的应用--堆排序 大家在学堆的时候,需要有二叉树的基础知识,大家可以看我的二叉树文章:二叉树 1.堆的实现 如果有⼀个关键码的集合 K {k0 , k1 , k2 , …,kn−1 } ,把它的所有元素按完全⼆叉树…...
新生讲课——图和并查集
1.图的存储 (1).邻接矩阵 邻接矩阵可以借助stl中的vector,我们通过开一个二维矩阵,g[u]中存储的是u可以到达的点,定义如下 const int N 2e5 10; vector<int> g[N] 若是遇到带权图则定义如下 const int N 2e5 10; vector <pair <int ,…...

基于深度学习的视觉检测小项目(十七) 用户管理后台的编程
完成了用户管理功能的阶段。下一阶段进入AI功能相关。所有的资源见文章链接。 补充完后台代码的用户管理界面代码: import sqlite3from PySide6.QtCore import Slot from PySide6.QtWidgets import QDialog, QMessageBoxfrom . import user_manage # 导入使用ui…...
实战:利用百度站长平台加速网站收录
本文转自:百万收录网 原文链接:https://www.baiwanshoulu.com/33.html 利用百度站长平台加速网站收录是一个实战性很强的过程,以下是一些具体的步骤和策略: 一、了解百度站长平台 百度站长平台是百度为网站管理员提供的一系列工…...

web-XSS-CTFHub
前言 在众多的CTF平台当中,作者认为CTFHub对于初学者来说,是入门平台的不二之选。CTFHub通过自己独特的技能树模块,可以帮助初学者来快速入门。具体请看官方介绍:CTFHub。 作者更新了CTFHub系列,希望小伙伴们多多支持…...

【C++】P1957 口算练习题
博客主页: [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: C 文章目录 💯前言💯题目描述输入格式:输出格式: 💯我的做法代码实现: 💯老师的做法代码实现: 💯对比分析&am…...

第二十三章 MySQL锁之表锁
目录 一、概述 二、语法 三、特点 一、概述 表级锁,每次操作锁住整张表。锁定粒度大,发生锁冲突的概率最高,并发度最低。应用在MyISAM、InnoDB、BDB等存储引擎中。 对于表级锁,主要分为以下三类: 1. 表锁 2. 元数…...

linux 进程补充
环境变量 基本概念 环境变量(environment variables)一般是指在操作系统中用来指定操作系统运行环境的一些参数 如:我们在编写C/C代码的时候,在链接的时候,从来不知道我们的所链接的动态静态库在哪 里,但是照样可以链接成功&#…...

渗透测试之文件包含漏洞 超详细的文件包含漏洞文章
目录 说明 通常分为两种类型: 本地文件包含 典型的攻击方式1: 影响: 典型的攻击方式2: 包含路径解释: 日志包含漏洞: 操作原理 包含漏洞读取文件 文件包含漏洞远程代码执行漏洞: 远程文件包含…...
[特殊字符] 智能合约中的数据是如何在区块链中保持一致的?
🧠 智能合约中的数据是如何在区块链中保持一致的? 为什么所有区块链节点都能得出相同结果?合约调用这么复杂,状态真能保持一致吗?本篇带你从底层视角理解“状态一致性”的真相。 一、智能合约的数据存储在哪里…...
内存分配函数malloc kmalloc vmalloc
内存分配函数malloc kmalloc vmalloc malloc实现步骤: 1)请求大小调整:首先,malloc 需要调整用户请求的大小,以适应内部数据结构(例如,可能需要存储额外的元数据)。通常,这包括对齐调整,确保分配的内存地址满足特定硬件要求(如对齐到8字节或16字节边界)。 2)空闲…...
多场景 OkHttpClient 管理器 - Android 网络通信解决方案
下面是一个完整的 Android 实现,展示如何创建和管理多个 OkHttpClient 实例,分别用于长连接、普通 HTTP 请求和文件下载场景。 <?xml version"1.0" encoding"utf-8"?> <LinearLayout xmlns:android"http://schemas…...
Golang dig框架与GraphQL的完美结合
将 Go 的 Dig 依赖注入框架与 GraphQL 结合使用,可以显著提升应用程序的可维护性、可测试性以及灵活性。 Dig 是一个强大的依赖注入容器,能够帮助开发者更好地管理复杂的依赖关系,而 GraphQL 则是一种用于 API 的查询语言,能够提…...
在四层代理中还原真实客户端ngx_stream_realip_module
一、模块原理与价值 PROXY Protocol 回溯 第三方负载均衡(如 HAProxy、AWS NLB、阿里 SLB)发起上游连接时,将真实客户端 IP/Port 写入 PROXY Protocol v1/v2 头。Stream 层接收到头部后,ngx_stream_realip_module 从中提取原始信息…...

江苏艾立泰跨国资源接力:废料变黄金的绿色供应链革命
在华东塑料包装行业面临限塑令深度调整的背景下,江苏艾立泰以一场跨国资源接力的创新实践,重新定义了绿色供应链的边界。 跨国回收网络:废料变黄金的全球棋局 艾立泰在欧洲、东南亚建立再生塑料回收点,将海外废弃包装箱通过标准…...
镜像里切换为普通用户
如果你登录远程虚拟机默认就是 root 用户,但你不希望用 root 权限运行 ns-3(这是对的,ns3 工具会拒绝 root),你可以按以下方法创建一个 非 root 用户账号 并切换到它运行 ns-3。 一次性解决方案:创建非 roo…...

【7色560页】职场可视化逻辑图高级数据分析PPT模版
7种色调职场工作汇报PPT,橙蓝、黑红、红蓝、蓝橙灰、浅蓝、浅绿、深蓝七种色调模版 【7色560页】职场可视化逻辑图高级数据分析PPT模版:职场可视化逻辑图分析PPT模版https://pan.quark.cn/s/78aeabbd92d1...

排序算法总结(C++)
目录 一、稳定性二、排序算法选择、冒泡、插入排序归并排序随机快速排序堆排序基数排序计数排序 三、总结 一、稳定性 排序算法的稳定性是指:同样大小的样本 **(同样大小的数据)**在排序之后不会改变原始的相对次序。 稳定性对基础类型对象…...
腾讯云V3签名
想要接入腾讯云的Api,必然先按其文档计算出所要求的签名。 之前也调用过腾讯云的接口,但总是卡在签名这一步,最后放弃选择SDK,这次终于自己代码实现。 可能腾讯云翻新了接口文档,现在阅读起来,清晰了很多&…...