【追求卓越11】算法--二叉树

引导

        接下来的几节我们开始介绍非线性的数据结构--树。树的内容比较多也比较复杂。本节,我们只需要了解关于树的一些基本概念。以及再进一步了解树的相关内容--搜索二叉树。该类型二叉树在工作中,是我们常接触的。该节我们介绍关于搜索二叉树的相关操作:查找,插入,删除。

什么是树?

关于树的定义:(摘自百度百科)

  1. 每个元素称为结点(node)
  2. 有一个特定的结点被称为根结点或树根(root)
  3. 除根结点之外的其余数据元素被分为m(m≥0)个互不相交的集合T1,T2,……Tm-1,其中每一个集合Ti(1<=i<=m)本身也是一棵树,被称作原树的子树(subtree)

说实话,这段定义我看了好几遍才能理解,对于刚接触的同学,结合图来理解是最好的了。

树中有很多需要我们了解的名词,我们结合下面的树来介绍:

        在上图中,A节点就是B节点的父节点,B节点是A节点的子节点。B,C,D这三个节点有共同的父节点,他们是兄弟节点。我们把没有父节点的节点称作为根节点,也就是图中的E节点。我们把没有子节点的的节点称作叶子节点或叶节点。

此外还有三个相似的概念:高度,深度,层。

节点的高度:节点到叶子节点最长路径。比如,上图中A节点的高度就是2.

节点的深度:根节点到节点的所经历的边数。比如,上图中B节点的深度是2.

节点的层:节点的深度+1。比如,上图中B子节点的层是3.

树的高度:根节点的高度。比如,上图中的树的高度是3.

以上就是关于树的部分名词定义了。但是我们工作中常接触的是二叉树。接下来,我们来正式进入今天的主题。

二叉树

        二叉树顾名思义,每个节点最多有两个分支,即仅有两个子节点。同理,有四个,八个分支的树就是四叉树,八叉树。是不是很容易理解?如图:

        上图中三个树都是二叉树,但是有两个树有比较特殊,需要注意。

        其中2号树中,叶子节点都在最低层,并且除了叶子节点,每个节点都有左右两个字节点。这种二叉树称作为满二叉树

        其中3号树中,叶子节点在最底下两层,最后一层的叶子节点都是靠左排列,并且除了最后一层,其他层的节点个数达到最大数,这种树称作完全二叉树。这个解释我觉得还不是很清晰,摘自百度百科:对于深度为K的,有n个结点的二叉树,当且仅当其每一个结点都与深度为K的满二叉树中编号从1n的结点一一对应时称之为完全二叉树

        到了这里,你可能会有疑问:满二叉树很容易理解,是二叉树的一种特例。但是完全二叉树存在的意义是什么

二叉树的存储方式

        我们常见的存储方式是基于指针或者引用的二叉链式存储法。他比较简单和直观,每个节点有一个数据段和两个左右指针。如图:

        另一种就是基于数组的顺序存储法。我们把根节点存储在下标i=1的位置,那左子节点存储的下标2 * i=2的位置,右子节点存储的下标2 * i+1的位置。

        根据这样的存储方式,刚刚的完全二叉树仅仅会浪费下标未0的空间,若不是完全二叉树就会出现浪费内存空间的现象。如图:

        所以对于完全二叉树,使用数组进行存储是最省内存的一种方式。因为顺序存储方式比链式存储方式更生内存,它不需要每个节点保存两个指针变量。

二叉树的遍历

对于二叉树的遍历,这个也是非常常见的面试题。常见的遍历方式是前序遍历,中序遍历,后序遍历。

前序遍历:对树中任意节点,先打印这个节点,然后打印它的左子树,最后打印右子树

中序遍历:对于树中任意节点,先打印这个节点的左子树,然后打印该节点,最后打印右子树

后序遍历:对于树中任意节点,先打印这个节点的左子树,然后打印右子树,最后打印改节点

对于遍历,实现方式也很简单,这里仅仅给出伪代码

void preOrder(Node* root) {
  if (root == null) return;
  print root //
此处为伪代码,表示打印root节点
  preOrder(root->left);
  preOrder(root->right);
}
void inOrder(Node* root) {
  if (root == null) return;
  inOrder(root->left);
  print root // 此处为伪代码,表示打印root节点
  inOrder(root->right);
}
void postOrder(Node* root) {
  if (root == null) return;
  postOrder(root->left);
  postOrder(root->right);
  print root // 此处为伪代码,表示打印root节点
}

每个节点最多被访问两次,故时间复杂度是O(n)。

搜索二叉树

        搜索二叉树又称为二叉查找树。顾名思义,二叉查找树是为了快速查找而诞生的。它不仅仅支持快速查找,还支持插入,删除。

定义:二叉查找树要求,在树中任意一个节点,其左子树中的每个节点的值,都要小于这个节点的值。而右子树节点的值都大于这个节点的值

可参考下图:

搜索二叉树的查找

  1. 先一个节点(一般是root节点),判断是否是我们要查找的数据?
  2. 如果要查找的数据比该节点的大,我们就在右子树中继续查找
  3. 如果要查找的数据比该节点的小,我们就在左子树中继续查找
  4. 若未查找到,则返回空

从上面的思路可以看出,我们可以用递归或者循环来实现,下面贴上代码(由于环境原因,代码都没有运行,不过代码逻辑应该是这样的):

typedef strcut node{
    int data;
    struct node left;
    struct node right;
}Node;
/
*while
循环*
/
Node * BinarySearchTree(Node* root , int target)
{
    Node * p = root;
    while(p != NULL)
    {
        if(p.data == target)
            return p;
        else if (p.data > target)
            p = p.left;
        else
            p = p.right;
    }
    return NULL;
}
/
*递归*
/
Node * BinarySearchTree(Node* root , int target)
{
    if(root == NULL)
        return NULL;

    if(root.data == target)
    {
        return root;
    }
    else if(root.data > target)
    {
        return BinarySearchTree(root.left,target);
    }
    else if(root.data < target)
    {
        return BinarysearchTree(root.right,target);
    }
}

  1. 搜索二叉树的插入

        插入操作和查找操作类似,新插入的数据一般都是保存在叶子节点上的。这句话,我刚开始是抱有怀疑态度,通过自己的思考也就理解了。建议大家先理解这句话的意思。

若知道了新插入数据都是保存在叶子节点上的,那么插入操作的思路也就简单了。

思路:

  1. 先获取一个节点,和插入数据比较哪个要大?
  2. 若要插入的数据比节点的值大,并且右子树为空,即直接将数据插入到该节点的右子节点中。若不为空,继续遍历右子树
  3. 若要插入的数据比节点的值小,并且左子树为空,即直接将数据插入到该节点的左子节点中。若部位空,继续遍历左子树

typedef strcut node{
    int data;
    struct node left;
    struct node right;
}Node;
int BinarySearchInsert(Node * root , int target)
{
    Node * p = root ;
    while(p != NULL)
    {
        if(p.data < target)
        {
            if(p.right == NULL)
            {
                p.right = newNode(target);
                return 0;
            }
            p = p.right;
        }
        else
        {
            if(p.left == NULL)
            {
                p.left = newNode(target);
                return 0;
            }
            p = p.left;
        }
    }
    return -1;
}

搜索二叉树的删除操作

搜索二叉树的删除操作就比较复杂了,因为删除之后,你必须要保证剩下的树结构满足搜索二叉树定义。其情况大致可以分为以下三种:

  1. 删除节点没有子节点。删除一个节点,我们需要先找到这个节点,我们可以使用查找操作的逻辑。若删除节点没有子节点,也就不会影响树的结构。直接删除即可。将其父节点指向该节点的指针,指向NULL;
  2. 删除的节点有一个子节点。将删除节点的父节点指向该节点的指针修改为指向子节点即可。
  3. 删除节点有两个子节点。将删除节点的右子树中最小节点删除,并替换为该删除节点即可。这个可能比较抽象,可参数下图,删除18这个节点:

typedef strcut node{
    int data;
    struct node left;
    struct node right;
}Node;
int BinarySearchDelete(Node * root , int target)
{
    Node * p = root;
    Node * father = NULL

    Node * minfather = NULL;
    Node * min = NULL;
    while(p != NULL && p.data != target)
    {
        father = p; /
*保存父节点*
/
        if(p.data < target)
            p = p.left;
        else
            p = p.right;
    }
    if(p == NULL)
        return -1;/
*not find target*
/

    /
*删除节点有两个子节点*
/
    if(p.left != NULL && p.right != NULL)
    {
        minfather = p;
        min = p.right;

        while(min.left != NULL )
            minfather = min;
            min = min.left;

        /
*至此找到删除节点右子树中的最小节点*
/
        p.data = min.data; /
*这里我觉得使用memcpy(p,min,sizeof(Node))较好*
/

        /
*将删除的点变为只有一个或没有子节点*
/
        p = min;
        father = minfather;
    }

    /
*至此,p表示删除的节点,father表示删除节点的父节点,并且删除节点不会有两个子节点*
/
    Node * child = NULL;
    if(p.left != NULL)
        child = p.left;
    else if(p.right != NULL)
        child = p.right;

    if(father == NULL)/
*删除的是root节点*
/
        memcpy(root,child,sizeof(Node));
    else if(father.left == p)
        father.left = child;
    else
        father.right = child;

    free(p);
}

搜索二叉树的中序遍历

        搜索二叉树有一个非常强大的特性,那就是通过中序遍历,即可输出有序的数据序列,时间复杂度是O(n),非常高效。

如何支持重复数据

上面的操作,我们都是假设没有重复数据,但是显示工作中我们总会遇到相同key的节点(key表示作为搜索二叉树依据)。针对相同键值的节点,我们又有哪些方式处理呢?

  1. 利用数组或链表进行扩容,将相同键值的数据储存到同一个节点上。
  2. 每个节点只保存一个数据。在插入的过程中,如果遇到一个节点的值,与要插入的值相同。我们就将这个要插入的数据放到这个节点的右子树。也就是将这个新插入的数据当作大于这个节点的值来处理。后续的删除都是按照这个逻辑处理

二叉查找树的时间复杂度

实际上二叉树的形态各式各样,即使同一组数据,也可以有不同的形态。如图:

第一种情况完全退化成了链表,时间复杂度为O(n),第三种的效果应该会好一些。其实,我们可以知道时间复杂度其实跟树的高度成正比。也就是O(height)。根据完全二叉树可知,L 的范围是[log2(n+1), log2n +1]。这样就极大的提高了效率。因此我们能在实际工作中需要无论怎么删除或增加,都可以保证树的平衡。红黑树就是这样的一种树。时间复杂度为O(logn)

搜索二叉树相较于散列表的优点

虽然散列表在的时间复杂度达到了O(1),而查找二叉树在理想情况下,也是O(logn)。那么为什么,我们工作中,树的使用要高于散列表呢?

  1. 散列表是无序存储,如果要输出有序的数据,需要先进行排序。而对于二叉树查找而言,我们仅需中序遍历就可以得到有序序列。
  2. 散列表扩容是耗时较多(虽然我们有一些优化手段,分时),而且当遇到散列冲突时,性能不稳定。虽然搜索二叉树也不稳定,但我们在工作中常使用的是平衡二叉查找树。
  3. 因为散列冲突的存在,散列表的时间复杂度并不一定比logn小。实际查找速度不一定比O(logn)快。另外还有哈希函数的耗时。
  4. 综上几点,树在某些方面还是优于散列表的。

总结

        本节开始接触树的内容,了解到父节点,子节点,兄弟节点,根节点,叶子节点,高度,深度,层等概念

        树的种类有很多,工作中,我们常接触的就是二叉树。我们也介绍了完全二叉树和满二叉树的概念。以及二叉树存储的方式有基于指针的链式存储方式和基于数组的顺序存储方式。其中完全二叉树使用数组是最节省内存的方式。

        绍了什么是搜索二叉树,以及搜索二叉树的删除,增加,查找的的逻辑。

        也简单介绍了,当有相同的键值时,我们该如何处理:利用数组或链表扩容。或者默认为大于该节点。

        最后我们讲解了二叉树的三种遍历方式:前序遍历,中序遍历,后序遍历。以及代码的是实现方式。树结构相对于散列表的优势。

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

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

相关文章

每日一题(LeetCode)----链表--链表最大孪生和

每日一题(LeetCode)----链表–链表最大孪生和 1.题目&#xff08;2130. 链表最大孪生和&#xff09; 在一个大小为 n 且 n 为 偶数 的链表中&#xff0c;对于 0 < i < (n / 2) - 1 的 i &#xff0c;第 i 个节点&#xff08;下标从 0 开始&#xff09;的孪生节点为第 (n…

广州华锐视点:基于VR元宇宙技术开展法律法规常识在线教学,打破地域和时间限制

随着科技的飞速发展&#xff0c;人类社会正逐渐迈向一个全新的时代——元宇宙。元宇宙是一个虚拟的、数字化的世界&#xff0c;它将现实世界与数字世界紧密相连&#xff0c;为人们提供了一个全新的交流、学习和娱乐平台。在这个充满无限可能的元宇宙中&#xff0c;法律知识同样…

【web】Fastapi自动生成接口文档(Swagger、ReDoc )

简介 FastAPI是流行的Python web框架&#xff0c;适用于开发高吞吐量API和微服务&#xff08;直接支持异步编程&#xff09; FastAPI的优势之一&#xff1a;通过提供高级抽象和自动数据模型转换&#xff0c;简化请求数据的处理&#xff08;用户不需要手动处理原始请求数据&am…

[vue3] 使用 vite 创建vue3项目的详细流程

一、vite介绍 Vite&#xff08;法语意为 “快速的”&#xff0c;发音 /vit/&#xff0c;发音同 “veet”) 是一种新型前端构建工具&#xff0c;能够显著提升前端开发体验&#xff08;热更新、打包构建速度更快&#xff09;。 二、使用vite构建项目 【学习指南】学习新技能最…

VM CentOS7安装ffmpeg

项目中涉及给视频添加水印&#xff0c;使用到了ffmpeg&#xff0c;windows系统可直接使用&#xff0c;Linux需要手动编译完成ffmpeg后才可正常使用。 配置yum源: 备份原repo文件 cd /etc/yum.repos.d/mv /etc/yum.repos.d/CentOS-Base.repo /etc/yum.repos.d/CentOS-Base.r…

详细解答T-SNE程序中from sklearn.manifold import TSNE的数据设置,包括输入数据,绘制颜色的参数设置,代码复制可用!!

文章目录 前言——TSNE是t-Distributed Stochastic Neighbor Embedding的缩写1、可运行的T-SNE程序2. 实验结果3、针对上述程序我们详细分析T-SNE的使用方法3.1 加载数据3.2 TSNE降维3.3 绘制点3.4 关于颜色设置&#xff0c;颜色使用的标签数据的说明cy 总结 前言——TSNE是t-D…

electron windows robotjs 安装教程

Robotjs 安装 前言第一步 : 安装python第二步 : 安装Visual Studio 2022第三步 : 安装robotjs 前言 robotjs可以控制鼠标键盘&#xff0c;获取屏幕内容&#xff0c;配合electron可做很多自动化操作。windows下配置环境有很多坑&#xff0c;很多文章都太旧了。试了很多次发现了…

【Java程序员面试专栏 专业技能篇】Java SE核心面试指引(三):核心机制策略

关于Java SE部分的核心知识进行一网打尽,包括四部分:基础知识考察、面向对象思想、核心机制策略、Java新特性,通过一篇文章串联面试重点,并且帮助加强日常基础知识的理解,全局思维导图如下所示 本篇Blog为第三部分:核心机制策略,子节点表示追问或同级提问 异常处理 …

软考:2024年软考高级:软件工程

软考&#xff1a;2024年软考高级: 提示&#xff1a;系列被面试官问的问题&#xff0c;我自己当时不会&#xff0c;所以下来自己复盘一下&#xff0c;认真学习和总结&#xff0c;以应对未来更多的可能性 关于互联网大厂的笔试面试&#xff0c;都是需要细心准备的 &#xff08;1…

docker compose搭建渗透测试vulstudy靶场示例

前言 渗透测试&#xff08;Penetration test&#xff09;即网络安全工程师/安全测试工程师/渗透测试工程师通过模拟黑客&#xff0c;在合法授权范围内&#xff0c;通过信息搜集、漏洞挖掘、权限提升等行为&#xff0c;对目标对象进行安全测试&#xff08;或攻击&#xff09;&am…

用Python写一个浏览器集群框架

更多Python学习内容&#xff1a;ipengtao.com 在分布式爬虫和大规模数据采集的场景中&#xff0c;使用浏览器集群是一种有效的方式&#xff0c;可以提高数据采集的速度和效率。本文将介绍如何用Python编写一个简单但强大的浏览器集群框架&#xff0c;以应对需要使用多个浏览器实…

5. 文件属性和目录

5. 文件属性和目录 1. Linux 系统的文件类型1.1 普通文件1.2 目录文件1.3 字符设备文件和块设备文件1.4 符号链接文件1.5 管道文件1.6 套接字文件 2. stat 系统调用2.1 struct stat 结构体2.2 st_mode 变量2.3 struct timespec 结构体 3. fstat 和 lstat 函数3.1 fstat 函数3.2…

机器学习---最大似然估计和贝叶斯参数估计

1. 估计 贝叶斯框架下的数据收集&#xff0c;在以下条件下我们可以设计一个可选择的分类器 : P(wi) (先验)&#xff1b;P(x | wi) (类条件密度) 但是。我们很少能够完整的得到这些信息! 从一个传统的样本中设计一个分类器&#xff1a; ①先验估计不成问题 ②对类条件密度…

Rust的Vec优化

本篇是对Rust编程语言17_Rust的Vec优化[1]学习与记录 MiniVec https://crates.io/crates/minivec enum DataWithVec { // tag,uint64,8字节 I32(i32), // 4字节,但需内存对齐到8字节? F64(f64), // 8字节 Bytes(Vec<u8>), // 24字节}fn main()…

Centos Bind安装与排错

1.配置Centos系统静态IP vi/etc/sysconfig/network-scripts/ifcfg-ens33BOOTPROTOstaticIPADDR192.168.1.100NETMASK255.255.255.0GATEWAY192.168.1.1DNS18.8.8.8:wqsudo systemctl restart network.service 2.安装BIND&#xff08;需要服务器连接互联网&#xff0c;如果服务…

基于springboot实现班级综合测评管理系统项目【项目源码+论文说明】计算机毕业设计

基于springboot实现班级综合测评管理系统演示 摘要 随着互联网技术的高速发展&#xff0c;人们生活的各方面都受到互联网技术的影响。现在人们可以通过互联网技术就能实现不出家门就可以通过网络进行系统管理&#xff0c;交易等&#xff0c;而且过程简单、快捷。同样的&#x…

git的创建以及使用

1、上传本地仓库 首先确定项目根目录中没有.git文件&#xff0c;有的话就删了&#xff0c;没有就下一步。在终端中输入git init命令。注意必须是根目录&#xff01; 将代码存到暂存区 将代码保存到本地仓库 2、创建git仓库 仓库名称和路径&#xff08;name&#xff09;随便写…

基于web宠颐生宠物医院系统设计与实现

基于web宠颐生医院系统开发与实现 摘要&#xff1a;时代飞速发展&#xff0c;网络也飞速发展&#xff0c;互联网许多的行业都可以用互联网实现了&#xff0c;互联网已经成为了人们生活中重要的一部分&#xff0c;或多或少的影响着我们的生活&#xff0c;互联网在给我带了方便的…

2021年8月18日 Go生态洞察:整合Go的网络体验

&#x1f337;&#x1f341; 博主猫头虎&#xff08;&#x1f405;&#x1f43e;&#xff09;带您 Go to New World✨&#x1f341; &#x1f984; 博客首页——&#x1f405;&#x1f43e;猫头虎的博客&#x1f390; &#x1f433; 《面试题大全专栏》 &#x1f995; 文章图文…

Linux环境下自动化创建大量的账号

参考《鸟哥的Linux私房菜基础篇第四版》13.7.2节微调而成&#xff1a; 下面脚本的目的是为服务器的管理员自动化创建大量的账号&#xff0c;节省生命。 #!/bin/bash # This shell script will create amount of Linux login accounts for you. # 1. check the "accounta…