二叉树的进阶

前言:

关于二叉树的基础知识,小生这里就不在一一一赘述了,对前面二叉树的基础知识有遗忘的铁子

们,可以康康前期咱的博客。

链接在此: 数据结构之二叉树 的精讲

目录:

一:二叉搜索树的定义

1.1 二叉搜索树的定义

首先二叉搜索树是一个递归的定义:左子树的所有节点都小于根节点;右子树 的所有节点都大于

根节点。

其中左子树,右子树又满足二叉搜索树 的性质

对于一个二叉搜索树而言,可以为空

对于一个二叉搜索树又名为二叉排序树或者二叉查找树。

对二叉搜索树进行中序遍历得到的是一个升序的序列。

中序遍历:先访问左孩子,再访问根节点,最后访问右孩子节点(递归的访问)

对于上面二叉搜索树进行输出:1   2   3    4    5    6   7

注意:一般而言,是不支持存储重复的节点。

二:二叉搜索树的实现

2.1 二叉搜索树的建立

其实对一棵二叉搜索树进行创建和对二叉树进行创建的过程基本是大同小异,只不过受到一些条件

的限制。

分析:

在插入指定的节点之后需要满足:左子树的节点小于根节点,右子树的节点大于根节点

前期的准备工作:

x:当前需要插入的节点

先找到一个合适的位置;在进行指针的链接

这里的 关键就是如何记录插入当期节点的父节点???

使用递归进行代码编写:在传参数的时候需要传当前节点的引用,因为当这是一颗空树 的时候,

在函数 的内部实现对树的修改,传引用直接解决了这一问题。

对应代码:

2.2 二叉搜索树的输出

其实就是一个中序的遍历;

这里借助递归实现中序遍历

之所嵌套一个函数实现中序的输出,是因为在函数外不能直接取到二叉搜索树的根节点,所以一

般嵌套一个子函数进行函数的调用

2.3 二叉搜索树的查找

依然是借助二叉搜索树的性质:左子树的所有节点小于根节点;右子树的所有 节点大于根节点

递归实现:

2.4二叉搜索树的删除

对于删除操作,没有前面那么简单了,难度就上来了~~~

1)删除叶子结点
2)删除只有一个孩子的节点

删除的是一个右孩子节点:

删除的是一个左孩子节节点:

 

通过画图发现对于删除叶节点或者删除只有一个孩子的节点,这两类情况可以归为一种问题

因为删除叶节点后,关键就是在于对应的父节点孩子到底是指向删除节点的左孩子还是右孩子的问

,此时不管指向左孩子也好,还是指向右孩子也罢,对于叶节点的孩子都是为空的,所以二者可

以归为一种情况。 

总体的删除框架:

1)先判断删除节点的孩子是否为空:目的避免对一个既有左孩子又有右孩子的节点进行删除

2)在判断删除的节点是否为根节点

3)最后进行判断删除的节点是父节点的左孩子还是右孩子:目的便于父节点与删除节点的孩子进

行链接

 对于判断删除节点是否为根节点还是删除节点是左孩子这一逻辑是不能进行颠倒的

 对应的代码:
//1)先对删除节点的孩子情况进行判断便于后期父节点孩子与删除节点的孩子进行链接if (cur->_left == nullptr){//2)必须先判断删除的节点是否为根节点:只有一个节点的二叉搜索树同时也是一个叶节点if (cur == _root){_root = _root->_right;delete cur;return true;}else{//3)再进行判断删除节点是左孩子还是右孩子if (cur == parent->_left){parent->_left = cur->_right;delete cur;return true;}else{parent->_right = cur->_right;delete cur;return true;}}}else if (cur->_right == nullptr){//这个整体的框架和上面删除一样//是否删除根节点if (cur == _root){_root = _root->_left;delete _root;return true;}else{// 删除的节点是左孩子还是右孩子if (parent->_left == cur){parent->_left = cur->_left;delete cur;return true;}else{parent->_right = cur->_left;delete cur;return true;}}}
3)删除有左右孩子的节点

 对应的代码:
//1)先找到右子树的最小节点:注意最小节点拷贝一定就是在左面:也可能最小节点是删除节点的右孩子TreeNode* subLeft = cur->_right;while (subLeft->_left){parent = subLeft;subLeft = subLeft->_left;}//找到了最小节点但是可能存在循环不进入的情况,所有parent可能为空,在初始化的时候parent= cur即可即可解决此问题//2)对_key  进行交换,目的转化为对最小节点的删除swap(cur->_key, subLeft->_key);//swap(key, subLeft->_key);//err// 3)判断subLeft是左孩子还是右孩子if (subLeft == parent->_left){parent->_left = subLeft->_right;delete subLeft;return true;}else{//parent->_right = subLeft->_left;//err这样写,当subLeft是叶节点的时候OK,但是当这是一个右叉树的时候,只能指向subLeft 的右孩子parent->_right = subLeft->_right;delete subLeft;return true;}}

对于容器一般进行的操作无非就是常见的“增删查改”。但是对于二叉搜索树进行修改操作后不一定

能保证修改之后还是一棵完整的二叉搜索树。

二叉树的常用函数:

拷贝构造函数:使用先序遍历进行创建

 对于拷贝构造函数的递归调用见下:

 这里有个小问题:对于参数为什么不能使用 引用传参,也就是 TreeNode* & root 

本质问题是:权限放大的问题

此时实参被const 进行修饰了,当在调用Copy(TreeNode* & root) 的时候,编译器会生成一个临时

对象,这个临时对象无法赋值给形参,因为权限被放大了。

注意:在传参的时候参数不是直接就传给形参了,而是编译器借助另一个临时变量,把实参赋值给

一个临时变量,再通过临时变量赋值给给形参,但是这样的效率比较慢。所以C++提出使用引用这

个概念,来解决此问题。

对于引用相关知识点,小生在这里就不在多多叨叨了,链接在此:引用相关介绍

赋值重载函数:

析构函数:

使用后续遍历进行销毁

接下来“硬菜”来了,使用递归编写对节点的删除代码:

 bool _EraseR(TreeNode* root, const k& key){//1:先对删除 的节点进行查找if (root->_key > key)_Erase(root->_left, key);else if (root->_key < key)_Erase(root->_right, key);else{//准备删除//2)判断删除节点左和右孩子是否为空if (root->_left == nullptr){//关键就是如何取到删除节点的父节点//参数:传引用TreeNode* del = root;root = root->_right;delete del;return true;}else if (root->_right == nullptr){TreeNode* del = root;root = root->_left;delete del;return true;}else{//删除节点的左右孩子都不为空//1:找右子树最小节点TreeNode* subLeft = root->_right;while (subLeft->_left){subLeft = subLeft->_left;}swap(root->_key, subLeft->_key);//转化为对subLeft所在的子树进行删除subLeft这个节点_Erase(root, key);return true;}}return false;}

总结:

以上就是关于二叉搜索树的基本操作,二叉搜索树是一种重要的数据结构,对于后面的AVL树,红

黑树,B树,B+树的学习,起着重要的作用。对于一个有序的数据的查询和输出,使用二叉搜索树

进行操作,效率是非常高的。

关于满二叉树,完全二叉树,二叉搜索树对比总结:

满二叉树:除了根节点和叶节点之外,所有的节点都有2个孩子节点,在高度为 h  的一个满二叉树

里面,总的节点个数是 2^h -1 

完全二叉树:是一种特殊的二叉树,假设高度为h,前  h-1 层符合满二叉树的特点;最后一次节点

可能满了也可能不满,但是一定是自左向右连贯的。

注意:满二叉树一定是完全二叉树,但是完全二叉树不一定就是满二叉树。

二叉搜索树:所有的左子树的键值都小于根节点,所有的右子树的键值都大于根节点(递归定

义);对二叉搜索树进行中序遍历输出的是一个升序的序列。

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

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

相关文章

从0开始linux(6)——gcc

欢迎来到博主的专栏&#xff1a;从0开始linux 博主ID&#xff1a;代码小豪、 文章目录 gccgcc的文件风格预处理编译汇编链接 gcc gcc是linux系统下常用的C语言编译器&#xff0c;随着后续的扩展&#xff0c;gcc支持了c&#xff0c;并推出了g编译器&#xff0c;现在的gcc可以支…

基于ssm疫情防控志愿者管理系统设计与实现

博主介绍&#xff1a;专注于Java&#xff08;springboot ssm springcloud等开发框架&#xff09; vue .net php phython node.js uniapp小程序 等诸多技术领域和毕业项目实战、企业信息化系统建设&#xff0c;从业十五余年开发设计教学工作 ☆☆☆ 精彩专栏推荐订阅☆☆☆…

轻松部署大模型:Titan Takeoff入门指南

轻松部署大模型&#xff1a;Titan Takeoff入门指南 在人工智能的快速发展中&#xff0c;处理自然语言处理&#xff08;NLP&#xff09;任务的大规模语言模型&#xff08;LLM&#xff09;至关重要。然而&#xff0c;部署这些模型往往具有挑战性&#xff0c;需要高性能的硬件和优…

论文(一)——寻找顶刊顶会

文章目录 一、顶刊二、顶会三、问题3.1 顶刊和顶会有什么区别3.1.1 定义3.1.2 评审流程3.1.3. 发表周期3.1.4 影响力与权威性3.1.5 适用领域3.1.6 交流与讨论 3.2 如何读论文 3.3 IEEE是啥&#xff1f;为什么这么多四、最后参考文章 一、顶刊 &#xff08;1&#xff09; IEEE …

《python语言程序设计》2018版第8章20题使用Rational类编写一个程序(上)-修改一下8-4Rational类我认为的错误

首先抄一下Rational类,可以安静的抄一遍 一、抄写中的问号 各种报错的截图1各种报错的截图2各种报错的截图3各种报错的截图4添加一个str我将n和d修改为self 书中214-215页间程序清单8-4的代码如下: class Rational:def __init__(self, numerator1, denominator0):divisor gcd(…

什么是 Tammann temperature

Tammann temperature (Tt_tt​) 是材料科学中一个重要的概念&#xff0c;它通常用于描述材料的热力学特性和相变行为。其定义与玻璃态和晶态材料的内部原子运动相关。Tammann 温度在研究材料的扩散、再结晶、以及玻璃化转变过程中具有重要意义。 1. Tammann 温度的定义 Tamma…

C语言实践: 使用哨兵找出数组中的最大元素

开篇 本题来源于《编程珠玑》第9章【代码调优】课后习题8。旨在实现一段使用哨兵找出数组中最大元素的逻辑代码。 题目描述 如何在程序中使用哨兵来找出数组中的最大元素? 思路分析 这个问题相对来说比较简单&#xff0c;以初始值作为哨兵&#xff0c;和后续的值进行比较及处理…

【目标检测】木制地板缺陷破损数据集338张6类VOC+YOLO格式

数据集格式&#xff1a;Pascal VOC格式YOLO格式(不包含分割路径的txt文件&#xff0c;仅仅包含jpg图片以及对应的VOC格式xml文件和yolo格式txt文件) 图片数量(jpg文件个数)&#xff1a;3383 标注数量(xml文件个数)&#xff1a;3383 标注数量(txt文件个数)&#xff1a;3383 标注…

最新网课搜题答案查询小程序源码/题库多接口微信小程序源码+自带流量主

源码简介&#xff1a; 最新网课搜题神器小程序源码&#xff0c;它是仿了小猿题库&#xff0c;功能多&#xff0c;能很快速找网课答案&#xff0c;还自带流量主功能。 这个小程序类似小助手&#xff0c;一键搜题就有答案。而且支持激励视频流量主&#xff0c;能轻松变现。 源…

iOS 18.1 將於 2024 年 10 月 28 日發布,並包含 Apple Intelligence 功能

在 9 月的活動中&#xff0c;Apple 發布了 iPhone 16 系列&#xff0c;Apple Intelligence 成為焦點功能。然而&#xff0c;最新的 iPhone 系列並未內建 Apple Intelligence 功能&#xff0c;這一點受到分析師和粉絲的廣泛批評。Apple 在活動中透露&#xff0c;Apple Intellige…

中国通信技术革命史

文章目录 引言I 中国通信技术革命史电报中国卫星通信的历史固定电话寻呼机(BP机)大哥大(手机)制定自己的移动通信网络技术体系5G未来科技发展的总趋势:用更少的能量,传输、处理和存储更多的信息II 知识扩展通信史(单位能量的信息传输率越来越高,网络地不断融合。)超级智能…

【C++】二叉搜索树+变身 = 红黑树

&#x1f680;个人主页&#xff1a;小羊 &#x1f680;所属专栏&#xff1a;C 很荣幸您能阅读我的文章&#xff0c;诚请评论指点&#xff0c;欢迎欢迎 ~ 目录 前言一、定义与性质二、红黑树节点的定义三、新增节点插入四、验证红黑树五、AVL树和红黑树比较 前言 本文仅适合了…

动态内存管理笔试题

目录 1.第一题1.1如何修改 2.第二题2.1题想2.2深刻理解 3.第三题4.第四题 1.第一题 void GetMemory(char* p) {p (char*)malloc(100); } void Test(void) {char* str NULL;GetMemory(str);strcpy(str, "hello world");printf(str); }请问运⾏Test 函数会有什么样的…

SSM湘农乐市农产品交易平台-计算机毕业设计源码28246

目 录 SSM湘农乐市农产品交易平台 1 绪论 1.1研究背景 1.2研究意义 1.3研究方法 1.4论文结构与章节安排 2 湘农乐市农产品交易平台系统分析 2.1 可行性分析 2.2 系统流程分析 2.3 系统功能分析 2.4 系统用例分析 2.5本章小结 3 湘农乐市农产品交易平…

环境对于写作有何影响?

如果你是有灵性、热爱文学创作的人&#xff0c;多半就会喜欢安静的生活环境。因为你会感受到唯有在这样的环境里更才能够沉下心来思考创作的路径。而且此时的你&#xff0c;显得头脑清醒、思维活跃而自由&#xff0c;因之文思泉涌。 网络图&#xff1a;宁静的书房 反之&#x…

【工作流引擎集成】springboot+Vue+activiti+mysql带工作流集成系统,直接用于业务开发,流程设计,工作流审批,会签

前言 activiti工作流引擎项目&#xff0c;企业erp、oa、hr、crm等企事业办公系统轻松落地&#xff0c;一套完整并且实际运用在多套项目中的案例&#xff0c;满足日常业务流程审批需求。 一、项目形式 springbootvueactiviti集成了activiti在线编辑器&#xff0c;流行的前后端…

Case:cocos地图和网格初始化

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言非盈利博客&#xff0c;只是学习笔记&#xff0c;如有雷同&#xff0c;十分抱歉。 一、生成一个100*100的网格背景代码分析导入必要的模块定义装饰器和类类定义…

c++继承(下)

c继承&#xff08;下&#xff09; &#xff08;1&#xff09;继承与友元&#xff08;2&#xff09;继承与静态成员&#xff08;3&#xff09;多继承及其菱形继承问题3.1 继承模型3.2 虚继承3.3 多继承中指针偏移问题 &#xff08;4&#xff09;继承和组合&#xff08;9&#xf…

Pytorch实现心跳信号分类识别(支持LSTM,GRU,TCN模型)

Pytorch实现心跳信号分类识别(支持LSTM,GRU,TCN模型&#xff09; 目录 Pytorch实现心跳信号分类识别(支持LSTM,GRU,TCN模型&#xff09; 1. 项目说明 2. 数据说明 &#xff08;1&#xff09;心跳信号分类预测数据集 3. 模型训练 &#xff08;1&#xff09;项目安装 &am…

大模型项目如何判断用RAG还是微调

大模型项目如何判断用RAG还是微调 在大模型项目中&#xff0c;选择使用检索增强生成&#xff08;Retrieval-Augmented Generation, RAG&#xff09;还是微调&#xff08;Fine-Tuning&#xff09;取决于多个因素&#xff0c;包括项目的具体需求、数据的可用性、性能要求、成本和…