Java学习苦旅(二十三)——二叉搜索树

2024-01-08 04:36

本文主要是介绍Java学习苦旅(二十三)——二叉搜索树,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本篇博客将详细讲解二叉搜索树。

文章目录

  • 二叉搜索树
    • 概念
    • 操作
      • 查找
      • 插入
      • 删除
    • 性能分析
  • 结尾

二叉搜索树

概念

二叉搜索树又称二叉排序树,它或者是一棵空树,或者是具有以下性质的二叉树:

若它的左子树不为空,则左子树上所有节点的值都小于根节点的值

若它的右子树不为空,则右子树上所有节点的值都大于根节点的值

它的左右子树也分别为二叉搜索树

image-20220314182534329

操作

查找

image-20220315145950439

示例代码

class Node {public int val;public Node left;public Node right;public Node(int val) {this.val = val;}
}public class BinarySearchTree {public Node root = null;public Node search(int key) {Node cur = root;while (cur != null) {if (cur.val < key) {cur = cur.right;} else if (cur.val == key) {return cur;} else {cur = cur.left;}}return null;}
}

插入

具体步骤:

  1. 如果树为空树,即根 == null,直接插入

  2. 如果树不是空树,按照查找逻辑确定插入位置,插入新结点

示例代码

public boolean insert(int val) {if (root == null) {root = new Node(val);return true;}Node cur = root;Node parent = null;while (cur != null) {if (cur.val < val) {parent = cur;cur = cur.right;} else if (cur.val == val) {return false;//不能有相同的数据} else {parent = cur;cur = cur.left;}}Node node = new Node(val);if (parent.val < val) {parent.right = node;} else {parent.left = node;}return true;
}

删除

设待删除结点为 cur, 待删除结点的双亲结点为parent

  1. cur.left == null
  • cur 是 root,则 root = cur.right

  • cur 不是 root,cur 是 parent.left,则 parent.left = cur.right

  • cur 不是 root,cur 是 parent.right,则 parent.right = cur.right

  1. cur.right == null
  • cur 是 root,则 root = cur.left

  • cur 不是 root,cur 是 parent.left,则 parent.left = cur.left

  • cur 不是 root,cur 是 parent.right,则 parent.right = cur.left

  1. cur.left != null && cur.right != null
  • 需要使用替换法进行删除,即在它的右子树中寻找中序下的第一个结点(关键码最小),用它的值填补到被删除节点中,再来处理该结点的删除问题

示例代码

public void remove(int key) {Node cur = root;Node parent = null;while (cur != null) {if (cur.val == key) {removeNode(cur,parent);break;}else if (cur.val < key) {parent = cur;cur = cur.right;} else {parent = cur;cur = cur.left;}}
}public void removeNode(Node cur, Node parent) {if (cur.left == null) {if (cur == root) {root = cur.right;} else if (cur == parent.left) {parent.left = cur.right;} else {parent.right = cur.right;}} 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;}} else {Node targetParent = cur;Node target = cur.right;while (target.left != null) {targetParent = target;target = target.left;}cur.val = target.val;if (target == targetParent.left) {targetParent.left = target.right;} else {targetParent.right = target.right;}}
}

性能分析

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

image-20220315170313260

最优情况下,二叉搜索树为完全二叉树,其平均比较次数为:log(N)

最差情况下,二叉搜索树退化为单支树,其平均比较次数为:N/2

结尾

本篇博客到此结束。
上一篇博客:Java学习苦旅(二十二)——Map&Set
下一篇博客:Java学习苦旅(二十四)——Java中的内部类

这篇关于Java学习苦旅(二十三)——二叉搜索树的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/582339

相关文章

Java利用Spire.XLS for Java设置Excel表格边框

《Java利用Spire.XLSforJava设置Excel表格边框》在日常的业务报表和数据处理中,Excel表格的美观性和可读性至关重要,本文将深入探讨如何利用Spire.XLSforJava库... 目录Spire.XLS for Java 简介与安装Maven 依赖配置手动安装 JAR 包核心API介

Java StringBuilder 实现原理全攻略

《JavaStringBuilder实现原理全攻略》StringBuilder是Java提供的可变字符序列类,位于java.lang包中,专门用于高效处理字符串的拼接和修改操作,本文给大家介绍Ja... 目录一、StringBuilder 基本概述核心特性二、StringBuilder 核心实现2.1 内部

SpringBoot AspectJ切面配合自定义注解实现权限校验的示例详解

《SpringBootAspectJ切面配合自定义注解实现权限校验的示例详解》本文章介绍了如何通过创建自定义的权限校验注解,配合AspectJ切面拦截注解实现权限校验,本文结合实例代码给大家介绍的非... 目录1. 创建权限校验注解2. 创建ASPectJ切面拦截注解校验权限3. 用法示例A. 参考文章本文

Java中字符编码问题的解决方法详解

《Java中字符编码问题的解决方法详解》在日常Java开发中,字符编码问题是一个非常常见却又特别容易踩坑的地方,这篇文章就带你一步一步看清楚字符编码的来龙去脉,并结合可运行的代码,看看如何在Java项... 目录前言背景:为什么会出现编码问题常见场景分析控制台输出乱码文件读写乱码数据库存取乱码解决方案统一使

Java Stream流与使用操作指南

《JavaStream流与使用操作指南》Stream不是数据结构,而是一种高级的数据处理工具,允许你以声明式的方式处理数据集合,类似于SQL语句操作数据库,本文给大家介绍JavaStream流与使用... 目录一、什么是stream流二、创建stream流1.单列集合创建stream流2.双列集合创建str

springboot集成easypoi导出word换行处理过程

《springboot集成easypoi导出word换行处理过程》SpringBoot集成Easypoi导出Word时,换行符n失效显示为空格,解决方法包括生成段落或替换模板中n为回车,同时需确... 目录项目场景问题描述解决方案第一种:生成段落的方式第二种:替换模板的情况,换行符替换成回车总结项目场景s

SpringBoot集成redisson实现延时队列教程

《SpringBoot集成redisson实现延时队列教程》文章介绍了使用Redisson实现延迟队列的完整步骤,包括依赖导入、Redis配置、工具类封装、业务枚举定义、执行器实现、Bean创建、消费... 目录1、先给项目导入Redisson依赖2、配置redis3、创建 RedissonConfig 配

SpringBoot中@Value注入静态变量方式

《SpringBoot中@Value注入静态变量方式》SpringBoot中静态变量无法直接用@Value注入,需通过setter方法,@Value(${})从属性文件获取值,@Value(#{})用... 目录项目场景解决方案注解说明1、@Value("${}")使用示例2、@Value("#{}"php

SpringBoot分段处理List集合多线程批量插入数据方式

《SpringBoot分段处理List集合多线程批量插入数据方式》文章介绍如何处理大数据量List批量插入数据库的优化方案:通过拆分List并分配独立线程处理,结合Spring线程池与异步方法提升效率... 目录项目场景解决方案1.实体类2.Mapper3.spring容器注入线程池bejsan对象4.创建

线上Java OOM问题定位与解决方案超详细解析

《线上JavaOOM问题定位与解决方案超详细解析》OOM是JVM抛出的错误,表示内存分配失败,:本文主要介绍线上JavaOOM问题定位与解决方案的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录一、OOM问题核心认知1.1 OOM定义与技术定位1.2 OOM常见类型及技术特征二、OOM问题定位工具