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

news2024/11/26 11:28:22

概述

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

之前说过,散列表也是支持这些操作的,并且散列表的这些操作比二叉查找树更高效,时间复杂度是 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.coloradmin.cn/o/1808851.html

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

相关文章

RAG:如何从0到1搭建一个RAG应用

通过本文你可以了解到&#xff1a; 什么是RAG&#xff1f;如何搭建一个RAG应用&#xff1f;目前开源的RAG应用有哪些&#xff1f; 大模型学习参考&#xff1a; 1.大模型学习资料整理&#xff1a;大模型学习资料整理&#xff1a;如何从0到1学习大模型&#xff0c;搭建个人或企业…

67. UE5 RPG 创建法师敌人角色

我们设置的敌人类型分三种&#xff0c;分别时战士类型&#xff0c;远程射手&#xff0c;和法师类型。在前面&#xff0c;我们创建了战士和射手类型的&#xff0c;还没有法师类型的&#xff0c;在这一篇文章中&#xff0c;我们创建一个法师类型的角色。 在64. UE5 RPG 创建新的双…

矩阵杯2024 Re wp 前两题

1. packpy UPX壳&#xff0c;但不能直接脱&#xff0c;应该是修改了头文件&#xff08;l_info) 改一下就能脱了 脱完是个elf文件 进IDA看一眼 明显的Py打包标志&#xff0c;用pyinstxtractor解包出来&#xff08;最好用对应的python3.8&#xff09; 可以得到packpy.pyc文件&a…

ROS学习记录:C++节点发布自定义地图

前言 ROS栅格地图格式 在了解了ROS地图消息包的数据结构后(链接在上)&#xff0c;本文将编写一个节点&#xff0c;发布地图消息包&#xff0c;看看在RViz中显示是什么效果。 一、准备 1、为了简单起见&#xff0c;发布一个两行四列的地图 2、为了便于观测&#xff0c;只对地…

textattack报错:不能导入自定义search_methods (cannot import name ‘xxx‘ from ‘xxx‘)

1. 报错信息 ImportError: cannot import name AAA from textattack.search_methods (/home/666/anaconda3/envs/textattack37_env/lib/python3.7/site-packages/textattack/search_methods/__init__.py)2. 出错简述 贴一段test1.py的模块导入 #建议使用&#xff01; import…

时钟影响ADC性能不仅仅是抖动

时钟影响ADC性能除了抖动&#xff0c;还有占空比。 在高速AD采样中&#xff0c;时钟占空比是非常重要的一个参数。时钟信号的上升沿控制ADC的采样&#xff0c;而下降沿控制着信号的保持&#xff0c;在一个周期内才可以完成量化输出&#xff0c;所以必须保持时钟的占空比为50%&…

Typora Markdown编辑器 for Mac v1.8.10 安装

Mac分享吧 文章目录 效果一、准备工作二、开始安装1、双击运行软件&#xff0c;将其从左侧拖入右侧文件夹中&#xff0c;等待安装完毕2. 应用程序显示软件图标&#xff0c;表示安装成功 三、运行调试1、修改主题2、显示文档列表&#xff0c;如下图3、查看版本信息 **安装完成&…

AI智能体做高考志愿填报分析

关注公众号&#xff0c;赠送AI/Python/Linux资料&#xff0c;对AI智能体有兴趣的朋友也可以添加一起交流 高考正在进行时&#xff0c;学生焦虑考试&#xff0c;家长们焦虑的则是高考志愿怎么填。毕竟一个好的学校&#xff0c;好的专业是进入社会的第一个敲门砖 你看张雪峰老师…

CvT(ICCV 2021)论文与代码解读

paper&#xff1a;CvT: Introducing Convolutions to Vision Transformers official implementation&#xff1a;https://github.com/microsoft/CvT 出发点 该论文的出发点是改进Vision Transformer (ViT) 的性能和效率。传统的ViT在处理图像分类任务时虽然表现出色&#xf…

xilinx的Aurora8B10B的IP仿真及上板测试(高速收发器十七)

前文讲解了Aurora8B10B协议原理及xilinx相关IP&#xff0c;本文讲解如何设置该IP&#xff0c;并且通过示例工程完成该IP的仿真和上板。 1、生成Aurora8B10B IP 如下图所示&#xff0c;首先在vivado的IP catalog中输入Aurora 8B10B&#xff0c;双击该IP。 图1 查找Aurora 8B10…

【Python】成功解决SyntaxError: invalid syntax

【Python】成功解决SyntaxError: invalid syntax 下滑即可查看博客内容 &#x1f308; 欢迎莅临我的个人主页 &#x1f448;这里是我静心耕耘深度学习领域、真诚分享知识与智慧的小天地&#xff01;&#x1f387; &#x1f393; 博主简介&#xff1a;985高校的普通本硕&am…

cuda学习笔记(3)

一 CPU和GPU的区别 衡量处理器优劣的重要的两个指标&#xff1a; 延时性&#xff1a;同量的数据&#xff0c;所需要的处理时间 吞吐性&#xff1a;处理速度不快&#xff0c;但是每次处理量很大 GPU设计理念是最大化吞吐量&#xff0c;使用很小的控制单元对应很小的内存 cpu的设…

类和对象(下+)_const成员、初始化列表、友元、匿名对象

类和对象&#xff08;下&#xff09; 文章目录 类和对象&#xff08;下&#xff09;前言一、const成员二、友元1.友元函数2.友元类 三、初始化列表四、explicit关键字五、匿名对象总结 前言 static成员、内部类、const成员、初始化列表、友元、匿名对象 一、const成员 将cons…

cleanmymac清理时要一直输入密码 CleanMyMac X一直提示输入密码的解决方案

CleanMyMac X是一款专业的Mac清理软件&#xff0c;可智能清理mac磁盘垃圾和多余语言安装包&#xff0c;快速释放电脑内存&#xff0c;轻松管理和升级Mac上的应用。同时CleanMyMac X可以强力卸载恶意软件&#xff0c;修复系统漏洞&#xff0c;一键扫描和优化Mac系统。 在使用Cle…

LeetCode | 2022.将一维数组转变为二维数组

这道题思路比较简单&#xff0c;比较容易想到的是先判断m和n构成的二维数组在形式上是否可以由原来的数组转变而成&#xff0c;若不可以返回空数组&#xff0c;若可以直接用一个二重循环遍历一遍即可&#xff0c;时间复杂度 O ( n 2 ) O(n^2) O(n2) class Solution(object):de…

数据结构初阶 · 链式二叉树的部分问题

目录 前言&#xff1a; 1 链式二叉树的创建 2 前序 中序 后序遍历 3 树的节点个数 4 树的高度 5 树的叶子节点个数 6 树的第K层节点个数 前言&#xff1a; 链式二叉树我们在C语言阶段已经实现了&#xff0c;这里介绍的是涉及到的部分问题&#xff0c;比如求树的高度&am…

三、安全工程练习题(CISSP)

1.三、安全工程练习题(CISSP)

找素数第二、三种方法

文章目录 第一种 &#xff1a;使用标签第二种&#xff1a;本质是方法的分装 第一种 &#xff1a;使用标签 没有使用信号量。break和continue作用范围只是最近的循环&#xff0c;无法控制外部循环。 此时使用标签 对外部循环进行操作。 package com.zhang; /* 找素数 第二种方…

【已解决】FileNotFoundError: [Errno 3] No such file or directory: ‘xxx‘

&#x1f60e; 作者介绍&#xff1a;我是程序员行者孙&#xff0c;一个热爱分享技术的制能工人。计算机本硕&#xff0c;人工制能研究生。公众号&#xff1a;AI Sun&#xff0c;视频号&#xff1a;AI-行者Sun &#x1f388; 本文专栏&#xff1a;本文收录于《AI实战中的各种bug…

【C语言】03.分支结构

本文用以介绍分支结构&#xff0c;主要的实现方式为if语句和switch语句。 一、if语句 1.1 if语句 if (表达式)语句表达式为真则执行语句&#xff0c;为假就不执行。在C语言中&#xff0c;0表示假&#xff0c;非0表示真.下图表示if的执行过程&#xff1a; 1.2 else语句 当…