【C++进阶】二叉搜索树

在这里插入图片描述

🚀write in front🚀
📜所属专栏: C++学习
🛰️博客主页:睿睿的博客主页
🛰️代码仓库:🎉VS2022_C语言仓库
🎡您的点赞、关注、收藏、评论,是对我最大的激励和支持!!!
关注我,关注我,关注我你们将会看到更多的优质内容!!

在这里插入图片描述

文章目录

  • 前言
  • 一.概念:
  • 二.二叉树的实现
    • 1.0版本(循环版本):
      • 1. 二叉搜索树的查找
      • 2. 二叉搜索树的插入
      • 3. 二叉搜索树的删除
    • 2.0版本(递归版本)
  • 三.搜索二叉树的应用:
  • 总结

前言

一.概念:

二叉搜索树又称二叉排序树,它或者是一棵空树,或者是具有以下性质的二叉树:

  • 若它的左子树不为空,则左子树上所有节点的值都小于根节点的值
  • 若它的右子树不为空,则右子树上所有节点的值都大于根节点的值
  • 它的左右子树也分别为二叉搜索树
    在这里插入图片描述

二.二叉树的实现

1.0版本(循环版本):

1. 二叉搜索树的查找

a、从根开始比较,查找,由二叉搜索树的性质可知,比根大则往右边走查找,比根小则往左边走查找。
b、最多查找高度次,走到到空,还没找到,这个值不存在。

2. 二叉搜索树的插入

插入的具体过程如下:
a. 树为空,则直接新增节点,赋值给root指针
b. 树不空,按二叉搜索树性质查找插入位置,插入新节点(这里的插入都是指的插在树叶的后面)

3. 二叉搜索树的删除

首先查找元素是否在二叉搜索树中,如果不存在,则返回, 否则要删除的结点可能分下面四种情况:
a. 要删除的结点无孩子结点
b. 要删除的结点只有左孩子结点
c. 要删除的结点只有右孩子结点
d. 要删除的结点有左、右孩子结点
看起来有待删除节点有4中情况,实际情况a可以与情况b或者c合并起来,因此真正的删除过程如下:

  1. 先找到要删除的结点和其父节点
  2. 找到之后分两种情况:
    a.删除结点有一个孩子或没有孩子:
  • 左为空:
    a1.父节点和删除结点如果重合,直接让root指向删除结点的右边

    a2父节点指向删除结点的右边
    在这里插入图片描述

  • 右为空(与上面同理)
    a3 父节点和删除结点如果重合,直接让root指向删除结点的左边
    a4父节点指向删除结点的左边

b.删除结点有两个孩子:
替换法删除:通过左子树的最大结点右树的最小结点来删除的结点替换,替换完就转化为a类的场景(左子树为空右子树为空),删掉替换后的结点即可,并且此时的那个结点一定有固定的一边是空的,由你选择的是左子树的最大结点还是右子树的最小结点而定。

在这里插入图片描述

总代码:

template<class T>struct BSTreeNode{BSTreeNode<T>* _left = nullptr;BSTreeNode<T>* _right = nullptr;T _key;BSTreeNode(const T& key) :_key(key){}};template<class T>class BStree{typedef BSTreeNode<T> Node;private:Node* _root = nullptr;void _InOder(Node* root){if (root == nullptr){return;}_InOder(root->_left);cout << root->_key << " ";_InOder(root->_right);}void _Destroy(Node* root){if (root == nullptr){return;}_Destroy(root->_left);_Destroy(root->_right);delete root;}Node* copy(Node* root){if (root == nullptr){return nullptr;}Node* copyroot = nullptr;copyroot = new Node(root->_key);copyroot->_left=copy(root->_left);copyroot->_right=copy(root->_right);return copyroot;}public:BStree(){}~BStree(){_Destroy(_root);}BStree(const BStree<T>& t){_root = copy(t._root);//在类里面可以看到私有的}BStree<T>& operator=(const BStree<T> t){swap(_root, t._root);return *this;}bool Insert(const T& key){if (_root == nullptr){_root = new Node(key);return true;}Node* cur = _root;Node* parent = _root;while (cur){if (key > cur->_key){parent = cur;cur = cur->_right;}else if (key < cur->_key){parent = cur;cur = cur->_left;}else{return false;}}cur = new Node(key);if (parent->_key > key){parent->_left = cur;}else if (parent->_key < key){parent->_right = cur;}return true;}bool Find(const T& key){Node* cur = _root;while (cur){if (key > cur->_key){cur = cur->_right;}else if (key < cur->_key){cur = cur->_left;}else{return true;}}return false;}void InOder(){_InOder(_root);cout << endl;}bool Erase(const T& key){Node* cur = _root;Node* parent = _root;while (cur){if (cur->_key < key){parent = cur;cur = cur->_right;}else if (cur->_key > key){parent = cur;cur = cur->_left;}//先寻找key的位置else{//找到之后有两种情况://第一种就是托孤:不管有一个孩子还是没有孩子都可以处理if (cur->_left == nullptr){//这里的坑就是删除头结点位置,此时cur和parent在同一个位置if (_root == cur){_root = cur->_right;}else{if (parent->_left == cur){parent->_left = cur->_right;}else{parent->_right = cur->_right;}}}else if (cur->_right == nullptr){//同样是上面那个坑if (_root == cur){_root = cur->_left;}else{if (parent->_left == cur){parent->_left = cur->_left;}else{parent->_right = cur->_left;}}}//第二种就是找到左树最大结点或右树最小结点,有两个孩子的时候else{//左树的最大结点:Node* _parent = cur;Node* LeftMax = cur->_left;while (LeftMax->_right){_parent = LeftMax;LeftMax = LeftMax->_right;}swap(LeftMax->_key, cur->_key);if (_parent->_right == LeftMax){_parent->_right = LeftMax->_left;}else{_parent->_left = LeftMax->_left;}cur = LeftMax;}delete cur;return true;}}return false;}};
}

2.0版本(递归版本)

  其实递归比循环好实现很多,只是存在栈溢出的问题罢了,并且递归有些地方是非常巧妙的,我们先来看看完整代码:

template<class T>struct BSTreeNode{BSTreeNode<T>* _left = nullptr;BSTreeNode<T>* _right = nullptr;T _key;BSTreeNode(const T& key) :_key(key){}};template<class T>class BStree{typedef BSTreeNode<T> Node;private:Node* _root = nullptr;void _InOder(Node* root){if (root == nullptr){return;}_InOder(root->_left);cout << root->_key << " ";_InOder(root->_right);}bool _InsertR(const T& key,Node*& root){if (root == nullptr){root = new Node (key);return true;}if (root->_key > key){return _InsertR(key, root->_left);}else if (root->_key < key){return _InsertR(key, root->_right);}else{return false;}}bool _FindR(const T& key, Node* root){if (root == nullptr){return false;}if (root->_key > key){return _FindR(key, root->_left);}else if (root->_key < key){return _FindR(key, root->_right);}else{return true;}}bool _EraseR(const T& key, Node*& root){if (root == nullptr){return false;}if (root->_key > key){return _EraseR(key, root->_left);}else if (root->_key < key){return _EraseR(key, root->_right);}else{Node* del = root;if (root->_right == nullptr){root = root->_left;}else if (root->_left == nullptr){root = root->_right;}else{Node* LeftMax = root->_left;while (LeftMax->_right){LeftMax = LeftMax->_right;}swap(LeftMax->_key, root->_key);_EraseR(key, root->_left);del = LeftMax;}delete del;return true;}}void _Destroy(Node* root){if (root == nullptr){return;}_Destroy(root->_left);_Destroy(root->_right);delete root;}Node* copy(Node* root){if (root == nullptr){return nullptr;}Node* copyroot = nullptr;copyroot = new Node(root->_key);copyroot->_left=copy(root->_left);copyroot->_right=copy(root->_right);return copyroot;}public:BStree(){}~BStree(){_Destroy(_root);}BStree(const BStree<T>& t){_root = copy(t._root);//在类里面可以看到私有的}BStree<T>& operator=(const BStree<T> t){swap(_root, t._root);return *this;}bool InsertR(const T& key){return _InsertR(key,_root);}bool FindR(const T& key){return _FindR(key,_root);}void InOder(){_InOder(_root);cout << endl;}bool EraseR(const T& key){return _EraseR(key, _root);}};
}

  我们来看一下递归在引用方面巧妙的使用:

这是循环的Insert
在这里插入图片描述
这是递归的_Insert
在这里插入图片描述
我们会明显发现递归在传参的时候传的是引用,为什么呢?接着往下看,循环的时候,为了使链接能够完成,我们会通过不断更新父节点来记录父节点,而递归的时候就不用。其实原因很显然,通过引用,在递归时的root就是他的父节点(上一个结点的root->rightroot->left)的别名,此时已经具有链接关系,在这里直接赋值就可以了。
在这里插入图片描述
在这里就省去了很多麻烦的事情,这就是递归的巧妙之处,而我们的循环不能使用引用,因为引用不能改变指向,在循环里面用引用只会使指向混乱,而在递归里面,会建立很多栈帧,每一个栈帧使用一次引用,在不出错的情况下还减少了保存父节点的麻烦。

同样的erase也是,在递归的时候使用引用,不仅减少了parent保存的麻烦,还省去了父节点和删除结点重合情况的特殊处理。

递归版本:
递归版本
循环版本:
在这里插入图片描述

三.搜索二叉树的应用:

  1. K模型:K模型即只有key作为关键码,结构中只需要存储Key即可,关键码即为需要搜索到的值。
    比如:给一个单词word,判断该单词是否拼写正确,具体方式如下:
  • 以词库中所有单词集合中的每个单词作为key,构建一棵二叉搜索树
  • 在二叉搜索树中检索该单词是否存在,存在则拼写正确,不存在则拼写错误。
  1. KV模型:每一个关键码key,都有与之对应的值Value,即<Key, Value>的键值对。该种方式在现实生活中非常常见:
    比如英汉词典就是英文与中文的对应关系,通过英文可以快速找到与其对应的中文,英文单词与其对应的中文<word, chinese>就构成一种键值对;
    再比如统计单词次数,统计成功后,给定单词就可快速找到其出现的次数,单词与其出
    现次数就是<word, count>就构成一种键值对
    还有就是我们平时刷身份证,平台通过我们身份证号码找到我们相关消息,只不过这个是更复杂的场景。下面我们来看一下具体的应用:

我们只需要把平常的多加一个value值即可,在插入的时候多插入一个值即可实现:

namespace key_value
{template<class T,class V>struct BSTreeNode{BSTreeNode<T,V>* _left = nullptr;BSTreeNode<T,V>* _right = nullptr;T _key;V _value;BSTreeNode(const T& key,const V& value) :_key(key),_value(value){}};template<class T,class V>class BStree{typedef BSTreeNode<T,V> Node;private:Node* _root = nullptr;void _InOder(Node* root){if (root == nullptr){return;}_InOder(root->_left);cout << root->_key << " "<<root->_value<<endl;_InOder(root->_right);}void Destroy(Node*& root){if (root == nullptr){return;}Destroy(root->_left);Destroy(root->_right);delete root;root = nullptr;}public:BStree(){}~BStree(){Destroy(_root);}bool Insert(const T& key,const V& value){if (_root == nullptr){_root = new Node(key,value);return true;}Node* cur = _root;Node* parent = _root;while (cur){if (key > cur->_key){parent = cur;cur = cur->_right;}else if (key < cur->_key){parent = cur;cur = cur->_left;}else{return false;}}cur = new Node(key,value);if (parent->_key > key){parent->_left = cur;}else if (parent->_key < key){parent->_right = cur;}return true;}Node* Find(const T& key){Node* cur = _root;while (cur){if (key > cur->_key){cur = cur->_right;}else if (key < cur->_key){cur = cur->_left;}else{return cur;}}return nullptr;}void InOder(){_InOder(_root);cout << endl;}bool Erase(const T& key){Node* cur = _root;Node* parent = _root;while (cur){if (cur->_key < key){parent = cur;cur = cur->_right;}else if (cur->_key > key){parent = cur;cur = cur->_left;}//先寻找key的位置else{//找到之后有两种情况://第一种就是托孤:不管有一个孩子还是没有孩子都可以处理if (cur->_left == nullptr){//这里的坑就是删除头结点位置,此时cur和parent在同一个位置if (_root == cur){_root = cur->_right;}else{if (parent->_left == cur){parent->_left = cur->_right;}else{parent->_right = cur->_right;}}}else if (cur->_right == nullptr){//同样是上面那个坑if (_root == cur){_root = cur->_left;}else{if (parent->_left == cur){parent->_left = cur->_left;}else{parent->_right = cur->_left;}}}//第二种就是找到左树最大结点或右树最小结点,有两个孩子的时候else{//左树的最大结点:Node* _parent = cur;Node* LeftMax = cur->_left;while (LeftMax->_right){_parent = LeftMax;LeftMax = LeftMax->_right;}swap(LeftMax->_key, cur->_key);if (_parent->_right == LeftMax){_parent->_right = LeftMax->_left;}else{_parent->_left = LeftMax->_left;}cur = LeftMax;}delete cur;return true;}}return false;}};
}

字典的例子:

void TestBSTree3()
{//BSTree<string, Date> carTree;key_value::BStree<string, string> dict;dict.Insert("insert", "插入");dict.Insert("sort", "排序");dict.Insert("right", "右边");dict.Insert("date", "日期");string str;while (cin >> str){key_value::BSTreeNode<string, string>* ret = dict.Find(str);if (ret){cout << ret->_value << endl;}else{cout << "无此单词" << endl;}}
}

统计个数的例子:

void TestBSTree4()
{// 11:44继续// 统计水果出现的次数string arr[] = { "西瓜", "西瓜", "苹果", "西瓜", "苹果", "苹果", "西瓜", "苹果", "香蕉", "苹果", "香蕉" };key_value::BStree<string, int> st;for (auto & str : arr){auto ret = st.Find(str);if (ret){ret->_value++;}else{st.Insert(str,1);}}st.InOder();
}

总结

  其实搜索二叉树就是将前面的数据结构和我们所学的C++进行了结合!其实我们不难看出,对于循环和递归,循环的使用就会伴随着很多特殊情况的出现,我们都要一一解决,并且不好读懂,但是他的性能很高,而递归则是很简短并且容易看懂,但是代价就是会出现栈溢出的现象。今后要根据实际情况具体选择。

  更新不易,辛苦各位小伙伴们动动小手,👍三连走一走💕💕 ~ ~ ~ 你们真的对我很重要!最后,本文仍有许多不足之处,欢迎各位认真读完文章的小伙伴们随时私信交流、批评指正!

专栏订阅:
每日一题
C语言学习
算法
智力题
初阶数据结构
Linux学习
C++学习
更新不易,辛苦各位小伙伴们动动小手,👍三连走一走💕💕 ~ ~ ~ 你们真的对我很重要!最后,本文仍有许多不足之处,欢迎各位认真读完文章的小伙伴们随时私信交流、批评指正!

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Android自动化测试之MonkeyRunner--从环境构建、参数讲解、脚本制作到实战技巧

monkeyrunner 概述、环境搭建 monkeyrunner环境搭建 (1) JDK的安装不配置 http://www.oracle.com/technetwork/java/javase/downloads/index.html (2) 安装Python编译器 https://www.python.org/download/ (3) 设置环境变量(配置Monkeyrunner工具至path目彔下也可丌配置) (4) …

【人工智能导论】线性回归模型

一、线性回归模型概述 线性回归是利用函数对一个或多个自变量和因变量之间关系进行建模的一种回归分析。简单来说&#xff0c;就是试图找到自变量与因变量之间的关系。 二、线性回归案例&#xff1a;房价预测 1、案例分析 问题&#xff1a;现在要预测140平方的房屋的价格&…

嵌入式Linux应用开发-基础知识-第十八章系统对中断的处理③

嵌入式Linux应用开发-基础知识-第十八章系统对中断的处理③ 第十八章 Linux系统对中断的处理 ③18.5 编写使用中断的按键驱动程序 ③18.5.1 编程思路18.5.1.1 设备树相关18.5.1.2 驱动代码相关 18.5.2 先编写驱动程序18.5.2.1 从设备树获得 GPIO18.5.2.2 从 GPIO获得中断号18.5…

Redis Cluster集群运维与核心原理剖析

文章目录 Redis集群方案比较哨兵模式高可用集群模式 Redis高可用集群搭建Java操作redis集群Redis集群原理分析槽位定位算法跳转重定位Redis集群节点间的通信机制集中式gossipgossip通信的10000端口 网络抖动Redis集群选举原理分析集群脑裂数据丢失问题集群是否完整才能对外提供…

Node18.x基础使用总结(二)

Node18.x基础使用总结 1、Node.js模块化1.1、模块暴露数据1.2、引入模块 2、包管理工具2.1、npm2.2、npm的安装2.3、npm基本使用2.4、搜索包2.5、下载安装包2.6、生产环境与开发环境2.7、生产依赖与开发依赖2.8、全局安装2.9、修改windows执行策略2.10、安装包依赖2.11、安装指…

深入了解 Linux 中的 AWK 命令:文本处理的瑞士军刀

简介 在Linux和Unix操作系统中&#xff0c;文本处理是一个常见的任务。AWK命令是一个强大的文本处理工具&#xff0c;专门进行文本截取和分析&#xff0c;它允许你在文本文件中查找、过滤、处理和格式化数据。本文将深入介绍Linux中的AWK命令&#xff0c;让你了解其基本用法和…

【多线程进阶】常见的锁策略

文章目录 前言1. 乐观锁 vs 悲观锁2. 轻量级锁 vs 重量级锁3. 自旋锁 vs 挂起等待锁4. 读写锁 vs 互斥锁5. 公平锁 vs 非公平锁6. 可重入锁 vs 不可重入锁总结 前言 本章节所讲解的锁策略不仅仅是局限于 Java . 任何和 “锁” 相关的话题, 都可能会涉及到以下内容. 这些特性主…

如何初始化一个vue项目

如何初始化一个vue项目 安装 vue-cli 后 ,终端执行 vue ui npm install vue-cli --save-devCLI 服务 | Vue CLI (vuejs.org) 等一段时间后 。。。 进入项目仪表盘 设置其他模块 项目构建后目录 vue.config.js 文件相关配置 官方vue.config.js 参考文档 https://cli.vuejs.o…

Linux基础指令(六)

目录 前言1. man 指令2. date 指令3. cal 指令4. bc 指令5. uname 指令结语&#xff1a; 前言 欢迎各位伙伴来到学习 Linux 指令的 第六天&#xff01;&#xff01;&#xff01; 在上一篇文章 Linux基本指令(五) 中&#xff0c;我们通过一段故事线&#xff0c;带大家感性的了…

BI神器Power Query(26)-- 使用PQ实现表格多列转换(2/3)

实例需求&#xff1a;原始表格包含多列属性数据,现在需要将不同属性分列展示在不同的行中&#xff0c;att1、att3、att5为一组&#xff0c;att2、att3、att6为另一组&#xff0c;数据如下所示。 更新表格数据 原始数据表&#xff1a; Col1Col2Att1Att2Att3Att4Att5Att6AAADD…

通过containerd部署k8s集群环境及初始化时部分报错解决

目录 一.基础环境配置&#xff08;每个节点都做&#xff09; 1.hosts解析 2.防火墙和selinux 3.安装基本软件并配置时间同步 4.禁用swap分区 5.更改内核参数 6.配置ipvs 7.k8s下载 &#xff08;1&#xff09;配置镜像下载相关软件 &#xff08;2&#xff09;配置kube…

8.3Jmeter使用json提取器提取数组值并循环(循环控制器)遍历使用

Jmeter使用json提取器提取数组值并循环遍历使用 响应返回值例如&#xff1a; {"code":0,"data":{"totalCount":11,"pageSize":100,"totalPage":1,"currPage":1,"list":[{"structuredId":&q…

nodejs+vue健身服务应用elementui

第三章 系统分析 10 3.1需求分析 10 3.2可行性分析 10 3.2.1技术可行性&#xff1a;技术背景 10 3.2.2经济可行性 11 3.2.3操作可行性&#xff1a; 11 3.3性能分析 11 3.4系统操作流程 12 3.4.1管理员登录流程 12 3.4.2信息添加流程 12 3.4.3信息删除流程 13 第四章 系统设计与…

使用docker完成minio服务部署扩容备份迁移生产实践文档

一、minio服务扩容方案 当服务器存储空间不足的时候&#xff0c;需要进行扩容&#xff0c;扩容过程中需要短暂停机时间&#xff0c;预计在一小时内能够完成和恢复 统一注意事项 强烈建议为部署中的所有节点选择基本相似的硬件配置。确保硬件&#xff08;CPU、内存、主板、存…

什么是物联网智慧公厕?

在当今科技快速发展的背景下&#xff0c;具备全感知、可靠传输、智能处理三大特点的物联网技术&#xff0c;正逐渐渗透到各个领域。而智慧公厕作为其中的一个创新应用&#xff0c;正逐渐受到市场的关注和重视。 什么是物联网智慧公厕&#xff1f;物联网智慧公厕是指通过物联网…

SmartX 边缘计算解决方案:简单稳定,支持各类应用负载

在《一文了解近端边缘 IT 基础架构技术需求》文章中&#xff0c;我们为大家分析了边缘应用对 IT 基础架构的技术要求&#xff0c;以及为什么超融合架构是支持边缘场景的最佳选择。值得一提的是&#xff0c;IDC 近日发布的《中国软件定义存储&#xff08;SDS&#xff09;及超融合…

Eclipse 主网即将上线迎空投预期,Zepoch 节点或成受益者?

目前&#xff0c;Zepoch节点空投页面中&#xff0c;模块化Layer2 Rollup项目Eclipse出现在其空投列表中。 配合近期Eclipse宣布了其将由SVM提供支持的Layer2主网架构&#xff0c;并将在今年年底上线主网的消息后&#xff0c;不免引发两点猜测&#xff1a;一个是Eclipse或将在不…

【数据代理+事件处理+计算属性与监视+绑定样式+条件渲染】

数据代理事件处理计算属性与监视绑定样式条件渲染 1 数据代理1.1 回顾Object.defineProperty方法1.2 数据代理 2 事件处理2.1 绑定监听2.2 事件修饰符2.3 键盘事件 3 计算属性与监视3.1 计算属性3.2 监视属性(侦视属性)3.3 watch对比computed 4 绑定样式4.1 绑定class样式4.2 绑…

[尚硅谷React笔记]——第2章 React面向组件编程

目录&#xff1a; 基本理解和使用&#xff1a; 使用React开发者工具调试函数式组件复习类的基本知识类式组件组件三大核心属性1: state 复习类中方法this指向&#xff1a; 复习bind函数&#xff1a;解决changeWeather中this指向问题&#xff1a;一般写法&#xff1a;state.htm…

进程之间的通信方式(共享存储,消息传递,管道通信)

进程通信 进程间通信&#xff08;Inter-Process Communication&#xff0c;IPC&#xff09;是指两个进程之间产生数据交互。进程是分配系统资源的单位(包括内存地址空间)&#xff0c;因此各进程拥有的内存地址空间相互独立。为了保证安全&#xff0c;一个进程不能直接访问另一…