C++二叉搜索树学习

news2024/9/21 13:55:53

目录

一、二叉搜索树概念

二、二叉搜索树的性能分析

三、二叉搜索树的构建


一、二叉搜索树概念

二叉搜索树又叫做二叉排序树,它可以是一颗空树,或者是具有以下性质的二叉树:

  • 若该树的左子树不为空,那么左子树上的任一节点都小于等于根节点的值。
  • 若该树的右子树不为空,那么右子树上的任一节点都大于等于根节点的值。
  • 该树的左右子树也为二叉搜索树。
  • 二叉搜索树可以支持插入相等的值,也可以不支持插入相等的值。

如上图,为一个基本的二叉搜索树。本文将以不插入相等的值的二叉搜索树为例。

二、二叉搜索树的性能分析

最优情况下,二叉搜索树为完全二叉树(或者接近完全二叉树),其高度为(log2 N),例如上方的二叉搜索树。

最差情况下,二叉搜索树退化为单支树(或者类似单支),其高度为(N/2 ~ N),例如:

综上所述,二叉搜索树的增删查改的时间复杂度为:O(N)。

三、二叉搜索树的构建

1. 基本框架

template<class K>
struct BSNode
{
	BSNode(const K& key)
		:_key(key)
		, _left(nullptr)
		, _right(nullptr)
	{}
	K _key;

	BSNode<K>* _left;
	BSNode<K>* _right;
};

template<class K>
class BSTree
{
public:
	using Node = BSNode<K>;
private:
	Node* _root = nullptr;
	int _num = 0;
};

我们以BSNode类,来构建二叉搜索树的每一个节点,这个节点有指向左右子树的指针_left和_right,并且每个节点都有一个值被_key所存储。

然后在BSTree类,是我们二叉搜索树的大框架,里面有着根节点_root,并且有着节点的数量_num。

2. 二叉搜索树的插入

bool _Insert(const K& key)
{
	Node* newnode = new Node(key);
	if (_root == nullptr)
	{
		_root = newnode;
		_num++;
		return true;
	}
	else
	{
		Node* parent = nullptr;
		Node* cur = _root;
		while (cur)
		{
			if (key < cur->_key)
			{
				parent = cur;
				cur = cur->_left;
			}
			else if (key > cur->_key)
			{
				parent = cur;
				cur = cur->_right;
			}
			else
				return false;
		}
		if (key < parent->_key)
		{
			_num++;
			parent->_left = newnode;
		}
		if (key > parent->_key)
		{
			_num++;
			parent->_right = newnode;
		}
		return true;
	}
	return false;
}

二叉搜索树的插入还是比较简单的,首先,我们要以传入的key值创建一个newnode的节点,如果根节点为空,那么直接把newnode给根节点,然后返回即可。

因为二叉搜索树的特殊性,所有左子树节点的值小于根节点,所有右子树节点的值大于根节点。因此当我们插入一个值(cur)时,首先要对到达的节点的值比较,如果跟该节点的值相等,因为我们讨论的是不支持有重复的值,返回false即可。

如果key小于该节点的值,那么我们就向该节点的左子树走,反之向该节点的右子树走,cur走到空,说明此处就是我们要插入的地方。因为我们提前已经创建parent指针来记录cur的父亲,所以此时我们只需要比较key的值和parent的值,然后决定插入在parent的左子树还是右子树即可。

3. 二叉树的查找

bool Find(const K& key)
{
	assert(_root != nullptr);

	Node* cur = _root;
	while (cur)
	{
		if (cur->_key < key)
		{
			cur = cur->_right;
		}
		else if (cur->_key > key)
		{
			cur = cur->_left;
		}
		else
		{
			return true;
		}
	}
	return false;
}

查找还是比较简单的,首先根节点不能为空,否则查找无意义。然后我们就和插入一样,从根节点开始,将我们要查找的key值与此时节点的值比较,key小就向节点的左子树走,key大就向节点的右子树走,若相等就返回true即找到了。如果cur都走到空了,那么就返回false即树中没有key值。

4. 二叉树的删除

二叉树的删除还是比较麻烦的,要分为以下四种情况:

  1. 该节点的左右子树均为空。
  2. 该节点的左子树为空,右子树不为空。
  3. 该节点的右子树为空,左子树不为空。
  4. 该节点的左右子树军不为空。
Node* cur = _root;
Node* parent = cur;
while (cur != nullptr && cur->_key != key)
{
	if (cur->_key < key)
	{
		parent = cur;
		cur = cur->_right;
	}
	else if (cur->_key > key)
	{
		parent = cur;
		cur = cur->_left;
	}	
}
//先找到要删除的cur节点,并记录该节点的父节点。

对于情况1,比较好处理,因为我们完全可以找到该节点,并记录该节点的父节点,然后直接删除该节点并且让父节点的对应指针指向空即可。

if (cur->_left == nullptr && cur->_right == nullptr)
{
	if (cur->_key < parent->_key)
		parent->_left = nullptr;
	else
		parent->_right = nullptr;
	delete cur;
	_num--;
	return true;
}

对于情况2,3我们可以同样地处理,以2为例。如果该节点的左子树为空,那么我们找到该节点后,并且记录了父节点,直接将父节点指向该节点的指针,指向该节点的右子树,然后直接删除该节点即可。情况3相同,只是注意左右子树不同。

else if(cur->_left == nullptr)
{
	Node* r_parent = cur;
	Node* r_root = cur->_right;
	while (r_root->_left)
	{
		r_parent = r_root;
		r_root = r_root->_left;
	}
	if (r_parent->_key > r_root->_key)
		r_parent->_left = r_root->_right;
	else if (r_parent->_key < r_root->_key)
		r_parent->_right = r_root->_right;
	cur->_key = r_root->_key;
	delete r_root;
	_num--;
	return true;
}
else if (cur->_right == nullptr)
{
	Node* l_parent = cur;
	Node* l_root = cur->_left;
	while (l_root->_right)
	{
		l_parent = l_root;
		l_root = l_root->_right;
	}
	if (l_parent->_key > l_root->_key)
		l_parent->_left = l_root->_left;
	else if (l_parent->_key < l_root->_key)
		l_parent->_right = l_root->_left;
	cur->_key = l_root->_key;
	delete l_root;
	_num--;
	return true;
}

对于情况4,我们不能直接删除该节点,因为该节点左右子树均不为空。我们此时就可以使用替换法。找到该节点(N)左子树的最大节点(L)或者右子树的最小节点(R)来替换该节点的值,因为这两个节点中的任意一个,放到N的位置,都不会破坏整个二叉搜索树的性质,然后我们删除交换后废弃的节点即可。例如:

我们要删除3这个节点,3的左子树根节点为1,右子树根节点为6,那么左子树最大的值为2,右子树最小的值为4,所以我们将2/4与3交换后不会改变整个二叉搜索树的性质。我们以交换4为例:

这样就实现了二叉搜索树的删除,也没有破坏本来的性质,注意如果4的右子树还有节点,那么交换后3的右子树就会有节点,那么我们只需要将6的左指针指向3的右子树即可。

else
{
	/*Node* parent = cur;
	Node* root = cur->_right;*/

	Node* r_parent = cur;
	Node* r_root = cur->_right;
	while (r_root->_left)
	{
		r_parent = r_root;
		r_root = r_root->_left;
	}
	if (r_parent->_key > r_root->_key)
		r_parent->_left = r_root->_right;
	else if (r_parent->_key < r_root->_key)
		r_parent->_right = r_root->_right;
	cur->_key = r_root->_key;
	delete r_root;
	_num--;
	return true;
}

注意,如果一开始根节点本来就为空,那么我们不需要删除,直接返回false即可,如果树只有一个节点,也就是_num为1,我们直接将_root给为nullptr即可:

if (cur == nullptr)
{
	return false;
}
if (_num == 1)
{
	_root = nullptr;
	_num = 0;
	return true;
}

整个删除的框架如下,最终只需将上述所以删除代码加入即可:

bool _Erase(const K& key)
{
    Node* cur = _root;
    Node* parent = cur;
    while (cur != nullptr && cur->_key != key)
    {
	    //查找要删除的节点位置,并且记录父节点
    }
    if (cur == nullptr)
    {
	    return false;//如果树为空或者没找到,直接返回false
    }
    if (_num == 1)
    {
	    //只有一个节点情况
    }
    if (cur->_left == nullptr && cur->_right == nullptr)
    {
	    //该节点左右子树均为空
    }
    else if(cur->_left == nullptr)
    {
	    //该节点左子树为空
    }
    else if (cur->_right == nullptr)
    {
	    //该节点右子树为空
    }
    else
    {
	    该节点左右子树均不为空
    }
	    return false;
}

以上内容如有错误,欢迎批评指正!!!

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

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

相关文章

硬件工程师笔试面试——存储器件

目录 16、存储器件 16.1 基础 存储器件实物图 16.1.1 概念 16.1.2 常见的存储器件及其特点 16.2 相关问题 16.2.1 不同类型的存储器件在成本和性能上有哪些具体的差异 16.2.2 如何根据应用需求选择合适的存储器件? 16.2.3 存储器件的耐用性和可靠性是如何影响其在不同…

【Unity】 HTFramework框架(五十六)MarkdownText:支持运行时解析并显示Markdown文本

更新日期&#xff1a;2024年9月15日。 Github源码&#xff1a;[点我获取源码] Gitee源码&#xff1a;[点我获取源码] 索引 MarkdownText支持的Markdown语法标题强调文本表格嵌入图像超链接 使用MarkdownText设置项运行时属性解析使用ID模式嵌入图像 MarkdownText MarkdownText…

【算法思想·二叉搜索树】基操篇

本文参考labuladong算法笔记[二叉搜索树心法&#xff08;基操篇&#xff09; | labuladong 的算法笔记] 1、概述 我们前文 东哥带你刷二叉搜索树&#xff08;特性篇&#xff09; 介绍了 BST 的基本特性&#xff0c;还利用二叉搜索树「中序遍历有序」的特性来解决了几道题目&am…

OpenAI的o1模型与Transformer的无限潜力:数学证明推理算力无上限

近期&#xff0c;斯隆奖得主马腾宇和Google Brain推理团队创始人Denny Zhou合作&#xff0c;提出了一项引人注目的数学证明&#xff1a;只要思维链&#xff08;CoT&#xff09;足够长&#xff0c;Transformer就有能力解决各种复杂问题。这一发现引发了广泛关注&#xff0c;因为…

驱动器磁盘未格式化难题:深度剖析与恢复实践

驱动器磁盘未格式化的深层探索 在数据存储与管理的日常中&#xff0c;驱动器作为我们数字生活的基石&#xff0c;其稳定性直接关系到数据的安全与可用性。然而&#xff0c;当屏幕上赫然出现“驱动器中的磁盘未被格式化”的提示时&#xff0c;许多用户往往感到手足无措&#xf…

把设计模式用起来!(3)用不好模式?之时机不对

上一篇&#xff1a;《把设计模式用起来&#xff08;2&#xff09;——用不好&#xff1f;之实践不足》 本篇继续讲设计模式用不好的常见原因&#xff0c;这是第二个&#xff1a;使用设计模式的时机不对。 二、时机不对 这里说的时机并不是单纯指软件研发周期中的时间阶段&…

C++11新增特性:lambda表达式、function包装器、bind绑定

一、lambda表达式 1&#xff09;、为啥需要引入lambda&#xff1f; 在c98中&#xff0c;我们使用sort对一段自定义类型进行排序的时候&#xff0c;每次都需要传一个仿函数&#xff0c;即手写一个完整的类。甚至有时需要同时实现排升序和降序&#xff0c;就需要各自手写一个类&…

基于SSM的社区爱心捐赠管理系统

作者&#xff1a;计算机学姐 开发技术&#xff1a;SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等&#xff0c;“文末源码”。 专栏推荐&#xff1a;前后端分离项目源码、SpringBoot项目源码、SSM项目源码 系统展示 【2025最新】基于JavaSSMVueMySQL的社区爱…

任嘉伦新剧《流水迢迢》:卫昭多层人设引关注

近日&#xff0c;由晋江文学城同名小说改编的武侠古装爱情传奇剧《流水迢迢》即将开播&#xff0c;这部由任嘉伦主演的新剧&#xff0c;在原著和阵容的双双加持下热度直线上涨&#xff0c;宣传阶段就已备受网友期待&#xff0c;预约人数截止9月13日已达到206万&#xff0c;上升…

通信工程学习:什么是GPON吉比特无源光网络

GPON&#xff1a;吉比特无源光网络 GPON&#xff08;Gigabit-Capable Passive Optical Network&#xff0c;吉比特无源光网络&#xff09;是一种基于ITU-T G.984.x标准的最新一代宽带无源光综合接入技术。该技术以其高带宽、高效率、大覆盖范围和用户接口丰富等特点&#xff0c…

ubuntu服务器版NVIDIA驱动失效解决方案

ubuntu服务器版NVIDIA驱动失效解决方案 1. 问题描述2. 解决方法--卸载并重新安装最新版显卡驱动cudacudnn2.1 卸载显卡驱动2.2 重新安装最新版显卡驱动cudacudnn2.2.1 显卡驱动2.2.2 cuda2.2.3 cuda安装cudnn 1. 问题描述 在终端输入nvidia-smi&#xff0c;输出如下&#xff1…

Leetcode—移除元素

移除元素 题目描述 思路 思路&#xff1a;定义两个指针变量指向数组第一个位置&#xff0c;判断nums[scr]是否等于val case1:相等&#xff0c;scr; case2:不相等&#xff0c;nums[dst]nums[scr],scr,dst; 时间复杂度&#xff1a;O&#xff08;n&#xff09;&#xff1b;空间复杂…

微信支付开发-后台统计工厂实现

一、数据库设计图 二、后端统计工厂逻辑 1、统计父抽象类 a、StatisticsHandle.php 2、统计工厂通道类 a、StatisticsFactory.php 3、查询实现类 a、答题统计(Answer.php) 三、后端统计工厂代码实现 1、统计父抽象类(StatisticsHandle.php) <?php /*** 统计父抽象类* Use…

基于密码的大模型安全治理的思考

文章目录 前言一、大模型发展现状1.1 大模型技术的发展历程1.2 大模型技术的产业发展二、大模型安全政策与标准现状2.1 国外大模型安全政策与标准2.2 我国大模型安全政策与标准前言 随着大模型技术的迅速发展和广泛应用,其安全性问题日益凸显。密码学作为网络空间安全的核心技…

Linux搭建邮箱服务器(简易版)

本章是上一文档的简易版本搭建方式更为快速简洁&#xff08;只需要两条命令即可搭建&#xff09;&#xff0c;如果想了解更详细一些可以看我上一文档 Linux接发邮件mailx_linux mailx o365-CSDN博客文章浏览阅读857次&#xff0c;点赞25次&#xff0c;收藏19次。本文详细描述了…

计算机组成原理-3.1储存系统

现代结构 1.储存器的层次结构 辅存的数据要调入主存后才能被CUP&#xff0c;与操作系统的进程进行联动 运行速度&#xff1a;CPU>寄存器>Cache>主存>磁盘>磁盘和光盘 主存-辅存:实现了虚拟系统&#xff0c;解决了主存容量不够的问题。 Cache-主存&#xff1a…

二叉树的前中后序遍历(递归法)( 含leetcode上三道【前中后序】遍历题目)

文章目录 深入理解递归思想递归三要素 leetcode上三道题目&#xff1a;144.二叉树的前序遍历145.二叉树的后序遍历94.二叉树的中序遍历 深入理解递归思想 这次我们要好好谈一谈递归&#xff0c;为什么很多同学看递归算法都是“一看就会&#xff0c;一写就废”。 主要是对递归…

宝塔部署python项目

宝塔部署-python项目文章浏览阅读559次&#xff0c;点赞11次&#xff0c;收藏9次。在添加项目后&#xff0c;选择项目所在的路径&#xff0c;然后命令行启动主py文件。具体先看项目日志&#xff0c;根据日志在环境管理处下载包。首先下载项目需要的python版本。_宝塔部署python…

Typora安装,使用,图片加载全流程!!!

文章目录 前言&#xff1a;安装&#xff1a;破解&#xff1a;使用typora&#xff1a;关于CSDN加载不出图片&#xff1a;创建OSS&#xff1a;设置PicGo&#xff1a; 前言&#xff1a; ​ Typora是一款非常流行的Markdown编辑器&#xff0c;简单来说就是可以方便我们写博客。拿我…

禁忌搜索算法(TS算法)求解实例---旅行商问题 (TSP)

目录 一、采用TS求解 TSP二、 旅行商问题2.1 实际例子&#xff1a;求解 6 个城市的 TSP2.2 **求解该问题的代码**2.3 代码运行过程截屏2.4 代码运行结果截屏&#xff08;后续和其他算法进行对比&#xff09; 三、 如何修改代码&#xff1f;3.1 减少城市坐标&#xff0c;如下&am…