数据结构与算法笔记:基础篇 - 二叉树基础(下):有了如此高效的散列表,为什么还需要二叉树?

概述

上篇文章,我们学习了树、二叉树及二叉树的遍历,本章来学习一种特殊的二叉树,二叉查找树。二叉查找树最大的特点就是,支持动态数据集合的快速插入、删除、查找操作。

之前说过,散列表也是支持这些操作的,并且散列表的这些操作比二叉查找树更高效,时间复杂度是 O ( 1 ) O(1) O(1)既然有了这么高效的散列表,使用二叉树的地方是不是都可以替换成散列表呢?有没有哪些地方时散列表做不了,必须要用二叉树来做的呢?

二叉查找树(Binary Search Tree)

二叉查找树的二叉树中最常用的一种类型,也叫二叉搜索树。顾名思义,二叉查找树是为了实现快速查找而生的。不过,它不仅仅支持快速查找一个数据,还支持快速插入、删除一个数据。它是怎么做到这些的呢?

这些都依赖于二叉查找树的特殊结构。二叉查找树要求,在树中的任意一个节点,其左子树中的每个 节点的值,都要小于这个节点的值,而右子树节点的值都大于这个节点的值。下图是二叉查找树的例子。

在这里插入图片描述

二叉查找树支持快速查找、插入、删除操作,现在我们就依次来看下,这三个操作是如何实现的。

1.二叉查找树的查找操作

首先,我们看如何在二叉查找树树中查找一个节点。我们先取根节点,如果它等于我们要查找的数据,那就返回。如果要查找的数据比根节点小,那就在左子树中递归查找;如果要查找的数据比根节点的值大,那就在右子树中递归查找。

在这里插入图片描述

下方是实现查找的代码。

public class BinarySearchTree {private Node tree;public Node find(int data) {Node p = tree;while (p != null) {if (data < p.data) p = p.left;if (data > p.data) p = p.right;return p;}return null;}public static class Node {private int data;private Node left;private Node right;public Node(int data) {this.data = data;}}
}

2.二叉查找树的插入操作

二叉查找树的插入过程有点类似查找操作。新插入的数据一班都是子啊叶子节点上,所以我们只需要从根节点开始,依次比较要插入的数据和节点的大小关系。

如果要插入的数据比节点的数据大,并且节点的右子树为空,就将新数据直接插入到右子节点的位置;如果不为空,就再递归遍历右子树,查找插入的位置。同理,如果要插入的数据比节点数值小,并且节点的左子树为空,就将新数据直接插入到左子节点的位置;如果不为空,就再递归遍历左子树,查找插入的位置。

在这里插入图片描述

下方是代码实现。

    public void insert(int data) {if (tree == null) {tree = new Node(data);return;}Node p = tree;while (p != null) {if (data > p.data) {if (p.right == null) {p.right = new Node(data);return;}p = p.right;} else {if (p.left == null) {p.left = new Node(data);return;}p = p.left;}}}

3.二叉查找树的删除操作

二叉查找树的查找、插入操作比较易懂,但是它的删除操作就比较复杂了。针对要删除的节点的子节点个数的不同,我们需要分三种情况来处理。

第一种情况是,如果要删除的节点没有子节点,我们只需要直接将父节点中,指向要删除结点地指针设置为 null。比如下图中的删除节点 55。

第二种情况是,如果要删除的子节点有一个子节点(只有左子节点或者右子节点),我们只需要更新父节点中,执行要删除的节点的指针,让它执行要删除的子节点就可以了。比如图中的删除结点 13。

第二种情况是,如果要删除的子节点有两个子节点,这就比较复杂了。我们需要找到这个节点的右子树中的最小节点,把它替换到要删除的节点上。然后再删除掉这个最小节点,因为最小节点肯定没有左子节点(如果有左子节点,那就不是最小节点了),所以,我们可以应用上面两条规则来删除这个最小节点。比如图中的删除节点 18。

在这里插入图片描述

老规矩,还是把删除的代码贴在这里。

    public void delete(int data) {Node p = tree; // p指向要哦删除的节点,初始化指向根节点Node pp = null; // pp记录的是p的父节点while (p != null && p.data != data) {pp = p;if (data > p.data) p = p.right;else p = p.left;}if (p == null) return; // 没有找到// 要删除的结点有两个子节点if (p.left != null && p.right != null) {Node minP = p.right;Node minPP = p; // minPP表示minP的父节点while (minP.left != null) {minPP = minP;minP = minP.left;}p.data = minP.data; // 将minP的数据替换到p中p = minP; // 从这开始,变成删除minP节点pp = minPP;}// 删除节点是叶子节点或者仅有一个子节点Node child;// p的子节点if (p.left != null) child = p.left;else if (p.right != null) child = p.right;else child = null;if (pp == null) tree = child; //删除的是根节点else if (pp.left == p) pp.left = child;else pp.right = child;}

实际上,关于二叉查找树的删除操作,还有个非常简单、取巧的办法,就是单纯将要删除的节点标记为 “已删除”,但是并不真正从树中将这个节点去掉。这样原本删除的结点还需要存储在内存中,比较浪费内存空间,但是删除操作就变得简单了很多。而且,这种处理方法也并没有增加插入、查找操作代码实现的难度。

4.二叉查找树的其他操作

除了插入、删除、查找操作外,二叉查找树中还可以支持快速查找最大节点和最小节点、前驱节点和后继节点。这些操作就不一一展示了。

二叉查找树除了支持上面几个操作之外,还有一个重要的特性,就是中序遍历二叉查找树,也可以输出有序的数据序列,时间复杂度是 O ( n ) O(n) O(n),非常高效。因此,二叉查找树也叫做二叉排序树。

支持重复数据的二叉查找树

前面讲二叉查找树的时候,默认树中节点存储的都是数字。很多时候,在实际开发中,我们在二叉查找树中存储的是,包含很多字段的对象。利用对象的某个字段作为键值(Key)来构建二叉查找树。我们把对象中的其他字段叫做卫星数据。

前面讲的二叉查找树的操作,针对的都是不存在键值相同的情况。那如果存储的两个对象键值相同,这种情况该怎么处理呢?这里有两种解决办法。

第一种方法比较容易。二叉查找树中每一个节点不仅会存储一个数据,因此,我们通过链表和支持动态扩容的数组等数据结构,把值相同的数据都存储在同一个节点上。

第二种方法比较不好理解,不过更加优雅。

每个节点仍然只存储一个数据。在查找插入位置的过程中,如果碰到一个节点的值,与要插入的数据的值相同,我们就将这个要插入的数据放到这个节点的右子树,也就是说,要把这个新插入的数据当做大于这个节点的值来处理。

在这里插入图片描述

当要查找数据的时候,遇到相同的节点,我们并不停止查找操作,而是继续在右子树中查找,直到遇到叶子节点,才停止。这样就可以把键值等于要查找值的所有节点都找出来。

在这里插入图片描述

对于删除操作,我们也需要查找到每个要删除的节点,然后再按前面讲的删除操作的方法,依次删除。

在这里插入图片描述

二叉查找树的时间复杂度分析

实际上,二叉查找树的形式各种各样。比如下图中,对于同一组数据,我们构造了三种二叉查找树。它们的查找、插入、删除操作的执行效率是不一样的。图中第一种二叉查找树,根节点放的左右子树极度不平衡,已经退化成了链表,所以查找的时间复杂度就变了了 O ( n ) O(n) O(n)

在这里插入图片描述

刚刚分析的是最糟糕的情况,现在来分析一下最理想的情况,二叉查找树是一棵完全二叉树(或者满二叉树)。这个时候,插入、删除、查找的时间复杂度是多少呢?

从前面的例子、图,以及还有代码来看,不管操作是插入、删除、还是查找,时间复杂度跟树的高度成成比,也就是 O(height)。既然这样,现在问题就变成了,如何求一棵包含 n 个节点额度二叉树的高度?

树的高度就等于最大层数减一,为了方便计算,我们转换成层来表示。从图中可以看出,包含 n 个接地那的完全二叉树中,第一层包含 1 个节点,第二层包含 2 个节点,第三层包含 4 个节点,依次类推,下一层节点个数是上一层的 2 倍,第 k 层包含的结点个数就是 2 k − 1 2^{k-1} 2k1

不过,对于完全二叉树来说,最后一层的节点个数有点不遵守上面的规律了。它包含的节点个数在 1 到 2 L − 1 2^{L-1} 2L1(假设最大层数是 L)。我们把每一层的节点个数加起来就是总的节点个数 n。也就是说,如果节点个数是 n,那么 n 满足这样一个关系:
n > = 1 + 2 + 4 + 8 + . . . + 2 L − 2 + 1 n >= 1 + 2 +4+8+...+2^{L-2} + 1 n>=1+2+4+8+...+2L2+1
n > = 1 + 2 + 4 + 8 + . . . + 2 L − 2 + 2 L − 1 n >= 1 + 2 +4+8+...+2^{L-2} + 2^{L-1} n>=1+2+4+8+...+2L2+2L1

借助等比数列的求和公式,我们可以计算出 L 的范围是 [ l o g 2 ( n + 1 ) log_2(n+1) log2(n+1), l o g 2 n + 1 log_2n + 1 log2n+1]。完全二叉树的层数小于等于 l o g 2 n + 1 log_2n + 1 log2n+1,也就是说,完全二叉树的高度小于等于 l o g 2 n log_2n log2n

显然,极度不平衡的二叉查找树,它的查找性能肯定不能满足我们的需求。我们需要构建一种不管怎么插入、删除数据,在任何时候,都能保持任意节点左右子树都比较平衡的二叉查找树,这就是下篇文章要讲解的,一种特殊的二叉查找树,平衡二叉树查找树。平衡二叉查找树的高度接近 l o g n logn logn,所以插入、删除、查找操作的时间复杂度也比较稳定,是 O ( l o g n ) O(logn) O(logn)

有了高效的散列表,为什么还需要二叉树?

我们在散列表那节讲过,散列表的插入、删除、查找操作的时间复杂度都可以做到常量级的 O ( 1 ) O(1) O(1),非常高效。而二叉查找树在比较平衡的情况下,插入、删除、查找操作的时间复杂度是 O ( l o g n ) O(logn) O(logn)。相对散列表,好像并没有什么优势,为什么还要用二叉查找树呢?

有以下几个原因:

第一,散列表中的数据是无序存储的,如果要输出有序数据,需要先进性排序。而对于二叉查找树来说,我们只需要中序遍历,就可以在 O ( n ) O(n) O(n) 的时间复杂度内,输出有序的数据序列。

第二,散列表扩容耗时很多,而且当遇到散列冲突时,性能不稳定,尽管二叉查找树的性能不稳定,但是在工程中,我们最常用的平衡二叉查找树的性能非常稳定,时间复杂度稳定在 O ( l o g n ) O(logn) O(logn)

第三,笼统地说,尽管散列表的查找等操作的时间复杂度是常量级的,但因为哈希冲突的存在,这个常量不一定比 logn 小,所以实际的查找速度可能不一定比 O ( l o g n ) O(logn) O(logn) 快。加上哈希函数的耗时,也不一定就比平衡二叉查找树查找的效率高。

第四,散列表的构造比二叉查找树要复杂,需要考虑的东西很多。比如散列函数的设计、冲突解决办法、扩容、缩容等。平衡二叉查找树只需要考虑平衡性这一个问题,而且这个问题的解决方案比较成熟、固定。

最后,为了避免过多的散列冲突,散列表装载因子不能太大,特别是基于开放寻址法解决冲突的散列表,不然解决散列冲突要花费一定的时间。

综合这几点,平衡二叉查找树在某些方面还是优于散列表的,所以,这两者的存储并不冲突。我们在实际的开发中需要结合具体需求来选择使用哪一个。

小结

本章学习了一种特殊的二叉树,二叉查找树。它支持快速地查找、插入、删除操作。

二叉查找树中,每个节点到的值都大于左子树节点的值,小于右子树节点的值。不过,这只是针对没有重复数据的情况。对于存在重复数据的二叉查找树,有两种构建方法:一种是让每个节点存储多个相同的数据;另一种是,湄公河节点中存储一个数据。针对这种情况,我们只需要稍加改造原来的插入、删除、查找操作即可。

在二叉查找树中,查找、插入、删除等很多操作的时间复杂度都跟树的高度成正比。两个极端情况的时间复杂度分别是 O ( n ) O(n) O(n) O ( l o g n ) O(logn) O(logn),分别对应二叉树退化成链表的情况和完全二叉树。

为了避免时间复杂度退化,针对二叉查找树,我们又设计了一种更加复杂的书,平衡二叉查找树,时间复杂度可以做到稳定的 O ( l o g n ) O(logn) O(logn),下一章节会具体讲解。

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

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

相关文章

盘点2024年5月Sui生态发展,了解Sui近期成长历程!

2024年5月是Sui的第一个生日月&#xff0c;Sui迎来了它的上线一周年纪念日。在过去的一年中Sui在技术进步与创新、生态系统的扩展、社区发展与合作伙伴关系以及重大项目和应用推出方面取得重要进展&#xff0c;展示了其作为下一代区块链平台的潜力。 以下是Sui的近期成长历程集…

QT 信号和槽 通过自定义信号和槽沟通 如何自定义槽和信号的业务,让它们自动关联 自定义信号功能

通过信号和槽机制通信&#xff0c;通信的源头和接收端之间是松耦合的&#xff1a; 源头只需要顾自己发信号就行&#xff0c;不用管谁会接收信号&#xff1b;接收端只需要关联自己感兴趣的信号&#xff0c;其他的信号都不管&#xff1b;只要源头发了信号&#xff0c;关联该信号…

STM32 | 独立看门狗 | RTC(实时时钟)

01、独立看门狗概述 在由单片机构成的微型计算机系统中,由于单片机的工作常常会受到来自外界电磁场的干扰,造成程序的跑飞,而陷入死循环,程序的正常运行被打断,由单片机控制的系统无法继续工作,会造成整个系统的陷入停滞状态,发生不可预料的后果,所以出于对单片机运行状…

Servlet与JSP的区别

Servlet和JSP&#xff08;JavaServer Pages&#xff09;都是Java EE&#xff08;Java Enterprise Edition&#xff09;规范的一部分&#xff0c;用于开发Web应用程序。它们在功能上有所重叠&#xff0c;但在设计和使用上有一些关键的区别&#xff1a; 1. 定义&#xff1a; …

QQ号码采集器-QQ邮箱采集器

寅甲QQ邮箱采集器或QQ号码采集软件, 一款采集QQ号、QQ邮件地址&#xff0c;采集QQ群成员、QQ好友的软件。可以按关键词采集&#xff0c;如可以按地区、年龄、血型、生日、职业等采集。采集速度非常快且操作很简单。

终于把tensorflow输入层和输出层搞懂了!fit函数与输入层,输出层,tf.keras.Model输入和输出的关系

结论 fit函数与输入层&#xff0c;输出层&#xff0c;tf.keras.Model输入和输出的关系 fit函数使用dataset格式&#xff0c;输入为字典格式&#xff0c;假设tf.keras.Model中输入和输出为字典格式&#xff08;2.2或2.3&#xff09;&#xff0c;dataset的key必须和2.2或2.3中字…

MySQL逻辑备份

目录 一.mysqldump 基本命令&#xff1a; 常用选项&#xff1a; 示例 备份整个数据库 备份多个数据库 备份所有数据库 仅备份数据库结构 仅备份特定表 添加选项以有效处理锁表问题 恢复数据库 从逻辑备份文件恢复 注意事项 二. mysqlpump mysqlpump 特点 基…

BoardLight - hackthebox

简介 靶机名称&#xff1a;BoardLight 难度&#xff1a;简单 靶场地址&#xff1a;https://app.hackthebox.com/machines/603 本地环境 靶机IP &#xff1a;10.10.11.11 ubuntu渗透机IP(ubuntu 22.04)&#xff1a;10.10.16.17 windows渗透机IP&#xff08;windows11&…

在 RISC-V 设计中发现可远程利用的漏洞

在移动CPU领域&#xff0c;主流的CPU构架除了intel 的X86构架&#xff0c;甲骨文的arm 构架&#xff0c;其实还有RISC-V 构架。但是因为国际间竞争关系&#xff0c;现在RISC-V技术路线被国外废止了&#xff0c;目前只有中国在继续开发&#xff08;早期RISC-V是买断过来的&#…

从欧盟弹性法案看软件物料清单(SBOM)

随着网络安全意识的提升和相关法规的推动&#xff0c;SBOM在国际上网络安全实践中的重要性日益凸显。 例如&#xff1a;美国国土安全部&#xff08;DHS&#xff09;的 “软件供应链评估工具包”&#xff08;SCAT&#xff09;就鼓励软件供应商提供SBOM&#xff0c;以帮助买方评…

重新认识Word —— 制作简历

重新认识Word —— 制作简历 PPT的图形减除功能word中的设置调整页边距进行排版表格使用 我们之前把word长排版文本梳理了一遍&#xff0c;其实word还有另外的功能&#xff0c;比如说——制作简历。 在这之前&#xff0c;我们先讲一个小技巧&#xff1a; PPT的图形减除功能 …

【数据结构】栈和队列-->理解和实现(赋源码)

Toc 欢迎光临我的Blog&#xff0c;喜欢就点歌关注吧♥ 前面介绍了顺序表、单链表、双向循环链表&#xff0c;基本上已经结束了链表的讲解&#xff0c;今天谈一下栈、队列。可以简单的说是前面学习的一特殊化实现&#xff0c;但是总体是相似的。 前言 栈是一种特殊的线性表&…

VISIO安装教程+安装包

文章目录 01、什么是VISIO&#xff1f;02、安装教程03、常见安装问题解析 01、什么是VISIO&#xff1f; Visio是由微软开发的流程图和图表绘制软件&#xff0c;它是Microsoft Office套件的一部分。Visio提供了各种模板和工具&#xff0c;使用户能够轻松创建和编辑各种类型的图…

【微信小程序开发(从零到一)】——个人中心页面的实战项目(二)

&#x1f468;‍&#x1f4bb;个人主页&#xff1a;开发者-曼亿点 &#x1f468;‍&#x1f4bb; hallo 欢迎 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! &#x1f468;‍&#x1f4bb; 本文由 曼亿点 原创 &#x1f468;‍&#x1f4bb; 收录于专栏&#xff1a…

VS2022+Qt雕刻机单片机马达串口上位机控制系统

程序示例精选 VS2022Qt雕刻机单片机马达串口上位机控制系统 如需安装运行环境或远程调试&#xff0c;见文章底部个人QQ名片&#xff0c;由专业技术人员远程协助&#xff01; 前言 这篇博客针对《VS2022Qt雕刻机单片机马达串口上位机控制系统》编写代码&#xff0c;代码整洁&a…

C#面:阐述对DDD的理解

C#是一种面向对象的编程语言&#xff0c;而领域驱动设计&#xff08;Domain-Driven Design&#xff0c;简称DDD&#xff09;是一种软件开发方法论&#xff0c;它强调将业务领域的知识和逻辑直接融入到软件设计和开发中。 在C#中实施DDD的关键是将业务领域划分为不同的领域模型…

PHP“well”运动健身APP-计算机毕业设计源码87702

【摘要】 随着互联网的趋势的到来&#xff0c;各行各业都在考虑利用互联网将自己的信息推广出去&#xff0c;最好方式就是建立自己的平台信息&#xff0c;并对其进行管理&#xff0c;随着现在智能手机的普及&#xff0c;人们对于智能手机里面的应用“well”运动健身app也在不断…

vue中插槽的本质

定义slotCompoent.vue 组件 <template><slot></slot><slot nameslot1></slot><slot name"slot2" msg"hello"></slot> </template>使用组件&#xff1a; <slotComponent><p>默认的</p>…

gcc:coverage:gcda文件没有生成的另一个例子:dlopen

根据gcc的文档&#xff0c; 如果是使用dlopen的方式来打开一个函数&#xff0c;需要记录coverage的数据&#xff0c;就需要使用下面这个链接。 If an executable loads a dynamic shared object via dlopen functionality, ‘-Wl,–dynamic-list-data’ is needed to dump all …

【系统架构】架构演进

系列文章目录 第一章 系统架构的演进 本篇文章目录 系列文章目录前言一、原始分布式二、单体系统时代三、SOA时代烟囱架构微内核架构事件驱动架构 四、微服务架构五、后微服务时代六、无服务时代总结 前言 最近笔者一直在学习系统架构的相关知识&#xff0c;对系统架构的演进…