【数据结构】Java实现二叉搜索树

news2025/1/11 14:11:10

二叉搜索树的基本性质

二叉搜索树(Binary Search Tree, BST)是一种特殊的二叉树,它具有以下特征:

1. 节点结构:每个节点包含一个键(key)和值(value),以及指向左子树和右子树的指针。

2. 左子树和右子树的性质:
   - 对于每个节点,左子树中所有节点的键都小于该节点的键。
   - 右子树中所有节点的键都大于该节点的键。
   - 这种特性使得二叉搜索树可以高效地进行查找、插入和删除操作。

3. 查找操作:从根节点开始,比较目标键与当前节点的键。如果目标键小于当前节点的键,则继续在左子树中查找;如果大于,则在右子树中查找。这个过程递归进行,直到找到目标节点或者到达树的叶子节点。

4. 插入操作:插入新节点时,同样从根节点开始,根据二叉搜索树的性质,选择左子树或右子树,直到找到一个空位置插入新节点。

5. 删除操作:删除节点较为复杂,分为三种情况:
   - 若被删除节点为叶子节点,直接移除。
   - 若被删除节点只有一个子节点,则用其子节点替代该节点。
   - 若被删除节点有两个子节点,则需要找到该节点的后继节点(通常是右子树中最小的节点),用其替代被删除的节点,并递归删除后继节点。

二叉搜索树的平均时间复杂度为O(log n),但在最坏情况下(如插入顺序导致树变为链表),其时间复杂度可达到O(n)。为了避免这一问题,通常会使用自平衡的二叉搜索树,如红黑树或AVL树。

二叉搜索树在许多应用中非常重要,包括数据库索引、内存中的排序数据以及许多算法实现等。

代码实现二叉搜索树的基本操作

查找操作

二叉搜索树(BST)中的查找操作是基于树的特性进行的,具体原理如下:

查找操作原理

  1. 初始化当前节点(cur):

    • 将当前节点设置为树的根节点(root),开始从树的顶端进行查找。
  2. 循环查找:

    • 使用while循环遍历树的节点,条件是cur不为null(即当前节点存在)。
  3. 比较当前节点与目标值:

    • 左子树查找:如果当前节点的值小于目标值,说明目标值可能在右子树中,因此将cur指向cur.right
    • 右子树查找:如果当前节点的值大于目标值,说明目标值可能在左子树中,将cur指向cur.left
    • 找到目标值:如果当前节点的值等于目标值,则查找成功,返回true
  4. 查找结束:

    • 如果循环结束,意味着没有找到目标值,此时返回false

代码实现:

public boolean search(int val) {
    // 初始化当前节点为根节点
    treeNode cur = root;

    // 当当前节点不为空时循环查找
    while (cur != null) {
        // 如果当前节点的值小于目标值,则在右子树继续查找
        if (cur.val < val) {
            cur = cur.right;
        // 如果当前节点的值大于目标值,则在左子树继续查找
        } else if (cur.val > val) {
            cur = cur.left;
        // 如果找到目标值,返回true
        } else {
            return true;
        }
    }
    // 如果遍历结束仍未找到目标值,返回false
    return false;
}

查找操作利用了二叉搜索树的排序性质,能够在相对较短的时间内找到目标值。通过比较和递归向左或向右子树移动,该操作在实际应用中被广泛使用,例如在需要快速检索数据的系统中,如数据库和内存中的索引结构等。

插入操作

二叉搜索树(Binary Search Tree, BST)的插入操作是基于树的特性进行的,以下是插入操作的原理详解:

插入操作原理

  1. 开始插入

    • 从树的根节点开始进行插入操作。
  2. 比较键值

    • 对于当前节点,比较要插入的值(key)与当前节点的键(val):
      • 如果要插入的值小于当前节点的键,则应该将其插入到左子树中。
      • 如果要插入的值大于当前节点的键,则应该将其插入到右子树中。
      • 如果要插入的值等于当前节点的键,通常视为重复值,根据具体需求,可以选择不插入、更新值或者其他处理。
  3. 寻找空位

    • 将当前节点更新为其左子树或右子树,然后重复该过程,直到找到一个空位置(即当前节点为null)。
    • 该空位置即为将新值插入树中的位置。
  4. 插入新节点

    • 在找到的空位置创建新的节点,并将其插入到树中。

 代码实现:

    public boolean insert(int val) {
        treeNode node = new treeNode(val);
        if(root == null) {
            root = node;
            return true;
        }
        treeNode cur = root;
        treeNode parent = null;
        while(cur != null) {
            if(val > cur.val) {
                parent = cur;
                cur = cur.right;
            }else if(val < cur.val) {
                parent = cur;
                cur = cur.left;
            }else {
                return false;
            }
        }
        if(val > parent.val) {
            parent.right = node;
        }else {
            parent.left = node;
        }
        return true;
    }

插入操作遵循了二叉搜索树的基本特性,通过比较和递归地前进来在合适的位置插入新节点。这一操作是保持树的结构的关键,确保每次插入后都能满足二叉搜索树的性质,以便后续的查找、删除等操作更加高效。

删除操作

  1. 查找要删除的节点

    • 从根节点开始,通过与当前节点的值比较,找到目标节点。如果目标值大于当前节点值则继续查找右子树;如果小于,则查找左子树。同时记录当前节点的父节点。
  2. 执行删除操作

    • 一旦找到要删除的节点,调用removeNode方法,根据要删除节点的子节点情况执行不同的删除逻辑。

代码实现:

public void remove(int key) {
    treeNode cur = root;        // 当前节点初始化为根节点
    treeNode parent = null;     // 记录当前节点的父节点

    // 找到要删除的节点
    while(cur != null) {
        if(key > cur.val) {
            parent = cur;       // 更新父节点
            cur = cur.right;    // 向右子树查找
        } else if(key < cur.val) {
            parent = cur;       // 更新父节点
            cur = cur.left;     // 向左子树查找
        } else {
            // 找到目标节点,执行删除逻辑
            removeNode(parent, cur);
            return;            // 找到并删除后退出
        }
    }
    System.out.println("没有该节点"); // 如果节点未找到,输出提示信息
}

// 执行删除操作的方法
private void removeNode(treeNode parent, treeNode cur) {
    // 情况1:要删除的节点没有左子节点
    if(cur.left == null) {
        if(cur == root) {
            root = cur.right;  // 如果要删除的节点是根节点,更新根节点
        } else if(cur == parent.left) {
            parent.left = cur.right; // 更新父节点的左子指针
        } else {
            parent.right = cur.right; // 更新父节点的右子指针
        }
    }
    // 情况2:要删除的节点没有右子节点
    else if(cur.right == null) {
        if(cur == root) {
            root = cur.left;   // 如果要删除的节点是根节点,更新根节点
        } else if(cur == parent.left) {
            parent.left = cur.left; // 更新父节点的左子指针
        } else {
            parent.right = cur.left; // 更新父节点的右子指针
        }
    }
    // 情况3:要删除的节点有两个子节点
    else {
        // 找到要删除节点的右子树中的最小节点
        treeNode target = cur.right;
        treeNode targetParent = cur;

        // 寻找右子树中最小节点
        while(target.left != null) {
            targetParent = target; // 更新最小节点的父节点
            target = target.left;   // 持续向左查找
        }

        // 用找到的最小节点的值替代要删除的节点的值
        cur.val = target.val;

        // 删除最小节点
        if(target == targetParent.right) {
            targetParent.right = target.right; // 更新父节点的右子指针
        } else {
            targetParent.left = target.right;  // 更新父节点的左子指针
        }
    }
}
  1. 查找节点

    • 使用while循环查找目标节点,记录其父节点,以便于后续的删除操作。比较节点值,并依据结果移动到左或右子树。
  2. 删除逻辑

    • 没有左子节点:直接将右子节点连接到父节点,删除当前节点。
    • 没有右子节点:直接将左子节点连接到父节点,同样删除当前节点。
    • 有两个子节点:找到右子树中最小的节点(后继节点),用其值替代当前节点的值,然后递归删除后继节点。
  3. 输出信息

    • 如果在树中找不到目标节点,输出“没有该节点”提示。

性能分析

插入和删除操作都必须先查找,查找效率代表了二叉搜索树中各个操作的性能。
对有n个结点的二叉搜索树,若每个元素查找的概率相等,则二叉搜索树平均查找长度是结点在二叉搜索树的深度的函数,即结点越深,则比较次数越多。
但对于同一个关键码集合,如果各关键码插入的次序不同,可能得到不同结构的二叉搜索树:

最优情况下,二叉搜索树为完全二叉树,其平均比较次数为:\log _{2}^{n}
最差情况下,二叉搜索树退化为单支树,其平均比较次数为:\frac{n}{2}

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

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

相关文章

大语言模型中的提示隐私保护

大语言模型中的提示隐私保护 一、简介&#xff1a; 大语言模型&#xff08;LLM&#xff09; 拥有庞大的规模、预先训练的知识和卓越的性能&#xff0c;被广泛应用于各种任务。提示学习(prompt learning)和指令微调(instruction tuning) 是两种重要的使得大模型能理解具体任务…

决策树基础

概述 决策树是一种树型结构&#xff0c;其中每个内部结点表示在一个属性上的测试&#xff0c;每个分支代表一 个测试输出&#xff0c;每个叶结点代表一种类别。决策树学习采用的是自顶向下的递归方法&#xff0c;其基本思想是以信息熵为度量构造一棵熵值下降最快的树&#xff…

自学Java第16Day

学习目标&#xff1a;面向对象进阶 学习内容&#xff1a;Object、Objects、BigInteger、BigDecimal、正则表达式 学习时间&#xff1a;下午 3 点-下午 6 点 学习产出&#xff1a; 1. Object类 1.1 概述 查看API文档&#xff0c;我们可以看到API文档中关于Object类的定义如下&…

【SpringBoot3】场景整合(实战)

0 环境准备 0.0 云服务器 阿里云、腾讯云、华为云 服务器开通&#xff1b; 按量付费&#xff0c;省钱省心 安装以下组件&#xff1a;docker、redis、kafka、prometheus、grafana 下载windterm&#xff1a; https://github.com/kingToolbox/WindTerm/releases/download/2.5…

React-Native 宝藏库大揭秘:精选开源项目与实战代码解析

1. 引言 1.1 React-Native 简介 React-Native 是由 Facebook 开发的一个开源框架&#xff0c;它允许开发者使用 JavaScript 和 React 的编程模型来构建跨平台的移动应用。React-Native 的核心理念是“Learn Once, Write Anywhere”&#xff0c;即学习一次 React 的编程模型&am…

《LeetCode热题100》---<5.普通数组篇六道>

本篇博客讲解LeetCode热题100道普通数组篇中的六道题 第一道&#xff1a;最大子数组和&#xff08;中等&#xff09; 第二道&#xff1a;合并区间&#xff08;中等&#xff09; 第一道&#xff1a;最大子数组和&#xff08;中等&#xff09; 法一&#xff1a;贪心算法 class So…

文件上传漏洞--之upload-labs靶场(第1关到第5关)专栏更新ing.....

第一关&#xff1a; 第一步&#xff1a;新建一个木马文件muma.php 第二步&#xff1a;在木马文件中写入一句话木马访问php主页 <?php eval(phpinfo());?> ​ 第三步&#xff1a;直接上传试一下 现上传一个png格式文件查看一下 然后上传一句话木马文件的&#xff0c;…

C:图案打印

引言 本篇文章讲了一些常见的图形编程题&#xff0c;并总结了一些规律。 1、打印空心正方形 1.1 代码展示&#xff1a; #include<stdio.h> int main() {int a 0;//边长初始化scanf("%d", &a);//输入边长的值{int i 0;for (i 0; i < a; i)//控制行…

面试经典算法150题系列-数组/字符串操作之多数元素

序言&#xff1a;今天是第五题啦&#xff0c;前面四题的解法还清楚吗&#xff1f;可以到面试算法题系列150题专栏 进行复习呀。 温故而知新&#xff0c;可以为师矣&#xff01;加油&#xff0c;未来的技术大牛们。 多数元素 给定一个大小为 n 的数组 nums &#xff0c;返回其…

Python 提取excel中嵌入的图片

注意&#xff1a;是嵌入在单元格里的图片&#xff0c;嵌入在单元格里的图片&#xff0c;嵌入在单元格里的图片 会显示这种类似的命令&#xff08;office好像不支持查看&#xff0c;wps可以查看这个位置显示的图片&#xff09; 网上好多都说用 _image 提取&#xff0c;经测试,…

精装房、旧房改造智能家居,单火线也有“救”了单火模块 零线发生器

精装房、旧房改造智能家居&#xff0c;单火线也有“救”了单火模块 零线发生器 史新华 以前写过关于智能家居没有预留零线&#xff0c;导致无法安装零火开关&#xff0c;也没办法装触控屏&#xff0c;主要原因还是无法通过零火线给设备供电。今年最火的一款思万奇零线发生器救…

LCM接口通讯说明

LCM&#xff08;Liquid Crystal Module&#xff0c;液晶模块&#xff09;接口通讯说明涉及多种接口类型和通讯方式&#xff0c;这些接口和通讯方式的选择取决于具体的应用场景和需求。 最常见的LCD模块接口协议是&#xff1a; 1.并行接口 2.串行接口 3.串行或并行配置到微处…

MySql的中的MVCC解决事务隔离性

MVCC 如何保证事务的隔离性&#xff1f; 1.排他锁&#xff1a;如一个事务获取了一个数据行的排他锁&#xff0c;其他事务就不能再获取改行的其他锁。 2.MVCC&#xff1a;多版本并发控制。 MVCC&#xff1a; 1.隐藏字段 1.DB_TRX_ID&#xff1a;最近修改事务的id。默认值从0开…

ShardingSphere-Jdbc + Spring Security + Redis 实现简单JWT认证

1. 项目结构 2. 数据库相关操作 create database user_profiles; use user_profiles; CREATE TABLE user (id INT AUTO_INCREMENT PRIMARY KEY,username VARCHAR(255) NOT NULL UNIQUE,password VARCHAR(255) NOT NULL,email VARCHAR(255) UNIQUE,role VARCHAR(2…

数据结构之《二叉树》(上)

在之前的数据结构的学习中&#xff0c;我们了解了顺序表、链表等线性表&#xff0c;接下来在本篇中将要学习一种非线性的数据结构——树&#xff0c;我们将来了解树的相关概念和性质&#xff0c;在树当中将重点学习二叉树的结构和特性。学习完相关概念后将试着实现二叉树&#…

我是怎么解决一个电力采集问题的全过程分享

行业设备联网&#xff0c;没人开发/开发太慢/投入太大 怎么办&#xff1f;用合宙DTU整体解决方案&#xff01; 一、整体解决方案内容 合宙DTU整体解决方案 DTU硬件&固件SIM卡业务云平台APP&小程序&web h5页面看板&#xff1b; 合宙提供的DTU整体解决方案&#x…

免费【2024】springboot 宠物中心信息管理系统app

博主介绍&#xff1a;✌CSDN新星计划导师、Java领域优质创作者、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和学生毕业项目实战,高校老师/讲师/同行前辈交流✌ 技术范围&#xff1a;SpringBoot、Vue、SSM、HTML、Jsp、PHP、Nodejs、Python、爬虫、数据可视化…

【超级宠物】-对战游戏

【超级宠物】是一款轻松好玩的休闲趣味卡通宠物模拟养成游戏,游戏中玩家需要不断的训练你的小宠物以参加比赛,你将会是一名专业的训宠物大师,成功的培养自己的宠物成为宠物之星。 超级宠物是一款充满趣味的休闲趣味宠物养成模拟竞技游戏&#xff0c;游戏中玩家需要培养你的狗狗…

Ollama0.3.0 + llama3.1本地部署

Ollama0.3.0 llama3.1本地部署 安装Ollama 官网下载安装包 https://ollama.com/download ​​ 根据系统选择对应版本 安装完成后&#xff0c;电脑右下角会出现ollama程序图标&#xff0c;表示ollama正在运行。 ​​ 打开cmd命令 下载Llama3.1 输入ollama&#xff0c…

51单片机嵌入式开发:22、STC89C52R控制 实现单总线温度传感器DS18b20的温度读取

STC89C52R控制 实现单总线温度传感器DS18b20的温度读取 1 概述1.1 介绍1.2 特点1.3 应用领域 2 DS18B20原理详解2.1 内部机理2.2 读写时序2.3 DS18B20操作代码程序 3 演示4 总结 配套演示例程 1 概述 DS18B20是一款数字温度传感器&#xff0c;由Maxim Integrated&#xff08;美…