数据结构-二叉搜索树(BST)

目录

什么是二叉搜索树

二叉搜索树的特性 

(1)顺序性

(2)局限性

二叉搜索树的应用 

二叉搜索树的操作

(1)查找节点

(2)插入节点

(3)删除节点

(4)中序遍历 


什么是二叉搜索树

        如图所示,二叉搜索树(binary search tree)满足以下条件。

  1. 对于根节点,左子树中所有节点的值 < 根节点的值 < 右子树中所有节点的值。
  2. 任意节点的左、右子树也是二叉搜索树,即同样满足条件 1. 

二叉搜索树

        二分搜索树有着高效的插入、删除、查询操作。

        平均时间的时间复杂度为 O(log n),最差情况为 O(n)。二分搜索树与堆不同,不一定是完全二叉树,底层不容易直接用数组表示故采用链表来实现二分搜索树。只有在高频添加、低频查找删除数据的场景下,数组比二叉搜索树的效率更高。

查找元素插入元素删除元素
普通数组O(n)O(n)O(n)
顺序数组O(logn)O(n)O(n)
二分搜索树O(logn)O(logn)O(logn)

二叉搜索树的特性 

(1)顺序性

        二分搜索树可以当做查找表的一种实现。

        我们使用二分搜索树的目的是通过查找 key 马上得到 value。minimum、maximum、successor(后继)、predecessor(前驱)、floor(地板)、ceil(天花板、rank(排名第几的元素)、select(排名第n的元素是谁)这些都是二分搜索树顺序性的表现。

(2)局限性

        二分搜索树在时间性能上是具有局限性的。

        在理想情况下,二叉搜索树是“平衡”的,这样就可以在 log⁡𝑛 轮循环内查找任意节点。

        然而,如果我们在二叉搜索树中不断地插入和删除节点,可能导致二叉树退化为如图所示的链表,相应的,二叉搜索树的查找操作是和这棵树的高度相关的,而此时这颗树的高度就是这颗树的节点数 n,这时各种操作的时间复杂度也会退化为 𝑂(𝑛) 。

二叉搜索树退化


二叉搜索树的应用 

  • 用作系统中的多级索引,实现高效的查找、插入、删除操作。
  • 作为某些搜索算法的底层数据结构。
  • 用于存储数据流,以保持其有序状态

二叉搜索树的操作

(1)查找节点

        给定目标节点值 num ,可以根据二叉搜索树的性质来查找。如图 7-17 所示,我们声明一个节点 cur ,从二叉树的根节点 root 出发,循环比较节点值 cur.val 和 num 之间的大小关系。

  • 若 cur.val < num ,说明目标节点在 cur 的右子树中,因此执行 cur = cur.right 。
  • 若 cur.val > num ,说明目标节点在 cur 的左子树中,因此执行 cur = cur.left 。
  • 若 cur.val = num ,说明找到目标节点,跳出循环并返回该节点。

bst_search_step4

        二叉搜索树的查找操作与二分查找算法的工作原理一致,都是每轮排除一半情况。循环次数最多为二叉树的高度,当二叉树平衡时,使用 𝑂(log⁡𝑛) 时间。示例代码如下:

/* 查找节点 */
TreeNode *search(BinarySearchTree *bst, int num) {TreeNode *cur = bst->root;// 循环查找,越过叶节点后跳出while (cur != NULL) {if (cur->val < num) {// 目标节点在 cur 的右子树中cur = cur->right;} else if (cur->val > num) {// 目标节点在 cur 的左子树中cur = cur->left;} else {// 找到目标节点,跳出循环break;}}// 返回目标节点return cur;
}

        通过查找节点的方法,我们可以完成98. 验证二叉搜索树 - 力扣(LeetCode)

/*** Definition for a binary tree node.* struct TreeNode {*     int val;*     struct TreeNode *left;*     struct TreeNode *right;* };*/
bool isValidBSTHelper(struct TreeNode* root, long min_val, long max_val) {// 如果根节点为空,直接返回 true,因为空树也是 BSTif (root == NULL) {return true;}// 检查当前节点值是否在范围内if (root->val <= min_val || root->val >= max_val) {return false;}// 递归检查左右子树,更新范围return isValidBSTHelper(root->left, min_val, root->val) && isValidBSTHelper(root->right, root->val, max_val);
}bool isValidBST(struct TreeNode* root) {// 使用 long 类型的最小值和最大值作为初始范围return isValidBSTHelper(root, LONG_MIN, LONG_MAX);
}

(2)插入节点

        给定一个待插入元素 num ,为了保持二叉搜索树“左子树 < 根节点 < 右子树”的性质,插入操作流程如图所示。

  1. 查找插入位置:与查找操作相似,从根节点出发,根据当前节点值和 num 的大小关系循环向下搜索,直到越过叶节点(遍历至 None )时跳出循环。
  2. 在该位置插入节点:初始化节点 num ,将该节点置于 None 的位置。

在二叉搜索树中插入节点

        在代码实现中,需要注意以下两点。

  • 二叉搜索树不允许存在重复节点,否则将违反其定义。因此,若待插入节点在树中已存在,则不执行插入,直接返回。
  • 为了实现插入节点,我们需要借助节点 pre 保存上一轮循环的节点。这样在遍历至 None 时,我们可以获取到其父节点,从而完成节点插入操作。

        代码范例如下,与查找节点相同,插入节点使用 𝑂(log⁡𝑛) 时间。

/* 插入节点 */
void insert(BinarySearchTree *bst, int num) {// 若树为空,则初始化根节点if (bst->root == NULL) {bst->root = newTreeNode(num);return;}TreeNode *cur = bst->root, *pre = NULL;// 循环查找,越过叶节点后跳出while (cur != NULL) {// 找到重复节点,直接返回if (cur->val == num) {return;}pre = cur;if (cur->val < num) {// 插入位置在 cur 的右子树中cur = cur->right;} else {// 插入位置在 cur 的左子树中cur = cur->left;}}// 插入节点TreeNode *node = newTreeNode(num);if (pre->val < num) {pre->right = node;} else {pre->left = node;}
}

        通过插入节点的方法,我们可以完成701. 二叉搜索树中的插入操作 - 力扣(LeetCode)

struct TreeNode* createTreeNode(int val) {struct TreeNode* ret = malloc(sizeof(struct TreeNode));// 设置节点值ret->val = val;// 左右子节点为空ret->left = ret->right = NULL;// 返回新创建的节点return ret;
}struct TreeNode* insertIntoBST(struct TreeNode* root, int val) {// 如果根节点为空,直接将新节点作为根节点返回if (root == NULL) {root = createTreeNode(val);return root;}// 定义游标节点为根节点struct TreeNode* pos = root;// 循环查找插入位置while (pos != NULL) {// 如果 val 小于当前节点值if (val < pos->val) {// 如果当前节点的左子节点为空,将新节点插入左子节点位置if (pos->left == NULL) {pos->left = createTreeNode(val);break;} else {// 否则继续向左子树查找插入位置pos = pos->left;}} else {// 如果 val 大于等于当前节点值// 如果当前节点的右子节点为空,将新节点插入右子节点位置if (pos->right == NULL) {pos->right = createTreeNode(val);break;} else {// 否则继续向右子树查找插入位置pos = pos->right;}}}// 返回根节点return root;
}

(3)删除节点

        先在二叉树中查找到目标节点,再将其删除。与插入节点类似,我们需要保证在删除操作完成后,二叉搜索树的“左子树 < 根节点 < 右子树”的性质仍然满足。因此,我们根据目标节点的子节点数量,分 0、1 和 2 三种情况,执行对应的删除节点操作。

        如图所示,当待删除节点的度为 0 时,表示该节点是叶节点,可以直接删除。

在二叉搜索树中删除节点(度为 0 )

        如下图所示,当待删除节点的度为 1 时,将待删除节点替换为其子节点即可。

在二叉搜索树中删除节点(度为 1 )

        当待删除节点的度为 2 时,我们无法直接删除它,而需要使用一个节点替换该节点。由于要保持二叉搜索树“左子树 < 根节点 < 右子树”的性质,因此这个节点可以是右子树的最小节点或左子树的最大节点

假设我们选择右子树的最小节点(中序遍历的下一个节点),则删除操作流程如下图所示。

  1. 找到待删除节点在“中序遍历序列”中的下一个节点,记为 tmp 。
  2. 用 tmp 的值覆盖待删除节点的值,并在树中递归删除节点 tmp 。

bst_remove_case3_step4

        删除节点操作同样使用 𝑂(log⁡𝑛) 时间,其中查找待删除节点需要 𝑂(log⁡𝑛) 时间,获取中序遍历后继节点需要 𝑂(log⁡𝑛) 时间。示例代码如下:

/* 删除节点 */
// 由于引入了 stdio.h ,此处无法使用 remove 关键词
void removeItem(BinarySearchTree *bst, int num) {// 若树为空,直接提前返回if (bst->root == NULL)return;TreeNode *cur = bst->root, *pre = NULL;// 循环查找,越过叶节点后跳出while (cur != NULL) {// 找到待删除节点,跳出循环if (cur->val == num)break;pre = cur;if (cur->val < num) {// 待删除节点在 root 的右子树中cur = cur->right;} else {// 待删除节点在 root 的左子树中cur = cur->left;}}// 若无待删除节点,则直接返回if (cur == NULL)return;// 判断待删除节点是否存在子节点if (cur->left == NULL || cur->right == NULL) {/* 子节点数量 = 0 or 1 */// 当子节点数量 = 0 / 1 时, child = nullptr / 该子节点TreeNode *child = cur->left != NULL ? cur->left : cur->right;// 删除节点 curif (pre->left == cur) {pre->left = child;} else {pre->right = child;}// 释放内存free(cur);} else {/* 子节点数量 = 2 */// 获取中序遍历中 cur 的下一个节点TreeNode *tmp = cur->right;while (tmp->left != NULL) {tmp = tmp->left;}int tmpVal = tmp->val;// 递归删除节点 tmpremoveItem(bst, tmp->val);// 用 tmp 覆盖 curcur->val = tmpVal;}
}

        除了迭代方法外,我们还可以使用递归方法来删除节点,下面的力扣题给出的方法就是递归方法。450. 删除二叉搜索树中的节点 - 力扣(LeetCode)

// 从二叉搜索树 root 中删除值为 key 的节点
struct TreeNode* deleteNode(struct TreeNode* root, int key) {// 如果根节点为空,直接返回 NULLif (root == NULL) {return NULL;}// 如果 key 小于当前节点值,递归删除左子树中的节点if (root->val > key) {root->left = deleteNode(root->left, key);return root;}// 如果 key 大于当前节点值,递归删除右子树中的节点if (root->val < key) {root->right = deleteNode(root->right, key);return root;}// 如果当前节点值等于 keyif (root->val == key) {// 如果当前节点没有左右子节点,直接返回 NULLif (!root->left && !root->right) {return NULL;}// 如果只有右子节点,返回右子节点if (!root->right) {return root->left;}// 如果只有左子节点,返回左子节点if (!root->left) {return root->right;}// 如果既有左子节点又有右子节点// 找到右子树中最小的节点作为当前节点的替代节点struct TreeNode *successor = root->right;while (successor->left) {successor = successor->left;}// 递归删除右子树中的最小节点root->right = deleteNode(root->right, successor->val);// 将替代节点的右子树连接到当前节点的右子树successor->right = root->right;// 将替代节点的左子树连接到当前节点的左子树successor->left = root->left;// 返回替代节点作为当前节点的父节点的子节点return successor;}// 返回根节点return root;
}

(4)中序遍历 

        如图所示,二叉树的中序遍历遵循“左 → 根 → 右”的遍历顺序,而二叉搜索树满足“左子节点 < 根节点 < 右子节点”的大小关系。

        这意味着在二叉搜索树中进行中序遍历时,总是会优先遍历下一个最小节点,从而得出一个重要性质:二叉搜索树的中序遍历序列是升序的。

        利用中序遍历升序的性质,我们在二叉搜索树中获取有序数据仅需 𝑂(𝑛) 时间,无须进行额外的排序操作,非常高效。

二叉搜索树的中序遍历序列

        利用二叉搜索树中序遍历升序,我们可以完成 530. 二叉搜索树的最小绝对差

void traversal(struct TreeNode* cur, struct TreeNode** pre, int *result) {if (cur == NULL) return;//BST中序遍历是升序traversal(cur->left, pre, result);   // 左if (*pre != NULL){       // 中*result = fmin(*result, cur->val - (*pre)->val);}*pre = cur; // 记录前一个traversal(cur->right, pre, result);  // 右
}int getMinimumDifference(struct TreeNode* root) {int result = 114514;struct TreeNode* pre = NULL; // 初始值为NULLtraversal(root, &pre, &result); // 传递pre的指针的指针return result;
}

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

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

相关文章

2024长三角快递物流展:科技激荡,行业焕发新活力

7月8日&#xff0c;杭州将迎来快递物流科技盛宴&#xff0c;这是一年一度的行业盛会&#xff0c;吸引了全球领先的快递物流企业和创新技术汇聚一堂。届时&#xff0c;会展中心将全方位展示快递物流及供应链、分拣系统、输送设备、智能搬运、智能仓储、自动识别、无人车、AGV机器…

【树莓派】yolov5 Lite,目标检测,行人检测入侵报警,摄像头绑定

延续之前的程序&#xff1a; https://qq742971636.blog.csdn.net/article/details/138172400 文章目录 播放声音pygame不出声音怎么办&#xff08;调节音量&#xff09;树莓派上的音乐播放器&#xff08;可选&#xff09;命令行直接放歌&#xff08;尝试放mp3歌曲&#xff09; …

深度学习的炼金术:转化数据为黄金的秘密

深度学习的炼金术&#xff1a;转化数据为黄金的秘密 1 引言 在现代深度学习的壮阔疆域中&#xff0c;数据是王冠上耀眼的宝石&#xff0c;而性能优化则是锻造这顶王冠的炼金术。这份融合了数据和算法魔力的艺术&#xff0c;不仅仅依赖于强大的计算资源和复杂的网络结构&#x…

Docker——开源的应用容器的引擎

目录 一、前言 1.虚拟化产品有哪些 1.1寄居架构 1.2源生架构 2.虚拟化产品对比/介绍 2.1虚拟化产品 2.1.1仿真虚拟化 2.1.2半虚拟化 2.1.3全虚拟化 2.2重点 2.2.1KVM——Linux内核来完成的功能和性能 2.2.2ESXI——用的比较多 二、Docker概述 1.Docker定义 2.Do…

常用算法代码模板 (3) :搜索与图论

AcWing算法基础课笔记与常用算法模板 (3) ——搜索与图论 常用算法代码模板 (1) &#xff1a;基础算法 常用算法代码模板 (2) &#xff1a;数据结构 常用算法代码模板 (3) &#xff1a;搜索与图论 常用算法代码模板 (4) &#xff1a;数学知识 文章目录 0 搜索技巧1 树与图的存…

【while循环】

目录 什么是循环 while语句的执行过程 编程求1*2*3*...*n 所有不超过1000的数中含有数字3的自然数 求数 求数II 编程求1平方2平方...n平方 什么是循环 循环就是重复做同样的事儿使用while语句循环输出1到100 int i 1; while( i < 100 ){cout <<…

Java虚拟机垃圾收集器详细总结

1、Serial收集器 Serial收集器是最基础、历史最悠久的收集器&#xff0c;曾经&#xff08;在JDK 1.3.1之前&#xff09;是HotSpot虚拟机新生代收集器的唯一选择。这个收集器是一个单线程工作的收集器&#xff0c;但它的“单线 程”的意义并不仅仅是说明它只会使用一个处理器或一…

自制贪吃蛇小游戏

此片文章涉及到到控制台设置的相关操作&#xff0c;虚拟键码&#xff0c;宽字符输出等&#xff0c;有些地方大家可能会看不懂&#xff0c;可以阅读以下文章来进一步了解&#xff1a; 控制台程序设置-CSDN博客 效果展示&#xff1a; QQ2024428-181932 源码已放在文章结尾 目录 …

Vivado-IP-DDS and Testbench Learning

DDS内部结构 实现流程 首先新建一个工程&#xff0c;创建bd文件&#xff0c;添加DDS Compiler核&#xff0c;此处不多赘述 Block Design 在观测输出的信号时&#xff0c;需要将最高位符号位的信号取反&#xff0c;这样才能输出正弦波&#xff0c;否则输出的波形如下图所示 将t…

Strassen矩阵乘法——C++

【题目描述】 根据课本“Strassen矩阵乘法”的基本原理&#xff0c;设计并实现一个矩阵快速乘法的工具。并演示至少10000维的矩阵快速乘法对比样例。 【功能要求】 实现普通矩阵乘法算法和“Strassen矩阵乘法”算法对相同的矩阵&#xff0c;分别用普通矩阵乘法算法&#xff…

C语言:数据结构(单链表)

目录 1. 链表的概念及结构2. 实现单链表3. 链表的分类 1. 链表的概念及结构 概念&#xff1a;链表是一种物理存储结构上非连续、非顺序的存储结构&#xff0c;数据元素的逻辑顺序是通过链表的指针链接次序实现的。 链表的结构跟火车车厢相似&#xff0c;淡季时车次的车厢会相应…

ULTIMATE VOCAL REMOVER V5 for Mac:专业人声消除软件

ULTIMATE VOCAL REMOVER V5 for Mac是一款专为Mac用户设计的人声消除软件&#xff0c;它凭借强大的功能和卓越的性能&#xff0c;在音乐制作和后期处理领域崭露头角。 ULTIMATE VOCAL REMOVER V5 for Mac v5.6激活版下载 这款软件基于深度神经网络&#xff0c;通过先进的训练模…

Windows使用bat远程操作Linux并执行命令

背景&#xff1a;让客户可以简单在Windows中能自己执行 Linux中的脚本&#xff0c;傻瓜式操作&#xff01; 方法&#xff1a;做一个简单的bat脚本&#xff01;能远程连接到Linux&#xff0c;并执行Linux命令&#xff01;客户双击就能使用&#xff01; 1、原先上网查询到使用P…

hadoop命令

hadoop命令 目录 hadoop命令 1.查看文件下面有哪些文件和目录 2.获取文件信息 查看文件内容 3.创建一个文件夹 4.剪切 1&#xff09;从本地hadoop剪切到hdfs并上传到hdfs 2&#xff09;剪切 从hdfs剪切到本地hadoop目录上 5.删除 1&#xff09;递归删除 2&#xff0…

AI Agent新对决:LangGraph与AutoGen的技术角力

AI Agent变革未来&#xff0c;LangGraph对抗AutoGen ©作者|Blaze 来源|神州问学 引言 比尔.盖茨曾在他的博客上发表一篇文章&#xff1a;《AI is about to completely change how you use computers》。在文章中&#xff0c;比尔盖茨探讨AI Agent对我们未来生活的巨大影…

windows电脑改造为linux

有个大学用的旧笔记本电脑没啥用了&#xff0c;决定把它改成linux搭一个服务器&#xff1b; 一、linux安装盘制作 首先要有一个大于8G的U盘&#xff0c;然后去下载需要的linux系统镜像&#xff0c;我下的是ubuntu&#xff0c;这里自选版本 https://cn.ubuntu.com/download/d…

请求响应案例-员工管理系统

准备工作 1、在pom.xml文件中引入dom4j依赖&#xff0c;解析xml文件。如果该依赖爆红&#xff0c;那么刷新以下就可以。 <!-- 解析XML --><dependency><groupId>org.dom4j</groupId><artifactId>dom4j</artifactId><version>2.1.3…

自然语言处理: 第三十章Hugging face使用指南之——trainer

原文连接: Trainer (huggingface.co) 最近在用HF的transformer库自己做训练&#xff0c;所以用着了transformers.Trainer&#xff0c;这里记录下用法 基本参数 class transformers.Trainer( model: Union None,args: TrainingArguments None,data_collator: Optional Non…

debian和ubuntu的核心系统和系统命令的区别

Debian和Ubuntu虽然有很深的渊源&#xff0c;都是基于Debian的发行版&#xff0c;但它们在核心系统和系统命令上还是有一些差别的。以下是一些主要的不同之处&#xff1a; 1. 发布周期&#xff1a; - Debian&#xff1a; Debian项目采用滚动发布模型&#xff0c;持续更新&a…

[论文笔记]GAUSSIAN ERROR LINEAR UNITS (GELUS)

引言 今天来看一下GELU的原始论文。 作者提出了GELU(Gaussian Error Linear Unit,高斯误差线性单元)非线性激活函数&#xff1a; GELU x Φ ( x ) \text{GELU} x\Phi(x) GELUxΦ(x)&#xff0c;其中 Φ ( x ) \Phi(x) Φ(x)​是标准高斯累积分布函数。与ReLU激活函数通过输入…