Java手写AVL树
Java手写AVL树
1. AVL树实现思路原理
为了解释AVL树的实现思路原理,下面使用Mermanid代码表示该算法的思维导图:
2. AVL树的手写必要性
手写AVL树的必要性主要体现在以下几个方面:
- 深入理解AVL树的原理和实现过程,加深对数据结构和算法的理解。
- 学习如何进行平衡二叉树的插入、删除和查找操作。
- 实际应用中可能需要自定义平衡二叉树的数据结构,手写AVL树可以满足特定需求。
3. AVL树的市场调查
市场调查显示,AVL树在各种领域都有广泛的应用。特别是在需要高效插入、删除和查找操作的场景下,AVL树具有较好的性能表现。常见的应用领域包括数据库索引、网络路由算法、编译器优化等。
4. AVL树的实现详细介绍和步骤
步骤1:定义AVL树的节点结构
首先,我们需要定义AVL树的节点结构,包括节点值、左子节点、右子节点、平衡因子等属性。
class AVLNode {int value;AVLNode left;AVLNode right;int balanceFactor;public AVLNode(int value) {this.value = value;this.left = null;this.right = null;this.balanceFactor = 0;}
}
步骤2:实现AVL树的插入操作
AVL树的插入操作是对二叉搜索树的插入操作进行了平衡调整。具体步骤如下:
- 在二叉搜索树中插入新节点。
- 更新插入路径上各节点的平衡因子。
- 如果某个节点的平衡因子绝对值大于1,则进行相应的平衡调整。
class AVLTree {private AVLNode root;// 插入节点public void insert(int value) {root = insertNode(root, value);}// 插入节点辅助函数private AVLNode insertNode(AVLNode node, int value) {if (node == null) {return new AVLNode(value);}if (value < node.value) {node.left = insertNode(node.left, value);} else if (value > node.value) {node.right = insertNode(node.right, value);} else {return node; // 值已存在,不插入}// 更新平衡因子node.balanceFactor = Math.max(getHeight(node.left), getHeight(node.right)) + 1;// 平衡调整int balance = getBalanceFactor(node);if (balance > 1 && value < node.left.value) {return rightRotate(node);}if (balance < -1 && value > node.right.value) {return leftRotate(node);}if (balance > 1 && value > node.left.value) {node.left = leftRotate(node.left);return rightRotate(node);}if (balance < -1 && value < node.right.value) {node.right = rightRotate(node.right);return leftRotate(node);}return node;}// 获取节点高度private int getHeight(AVLNode node) {if (node == null) {return 0;}return Math.max(getHeight(node.left), getHeight(node.right)) + 1;}// 获取节点的平衡因子private int getBalanceFactor(AVLNode node) {if (node == null) {return 0;}return getHeight(node.left) - getHeight(node.right);}// 左旋转private AVLNode leftRotate(AVLNode node) {AVLNode rightChild = node.right;AVLNode leftGrandChild = rightChild.left;rightChild.left = node;node.right = leftGrandChild;node.balanceFactor = Math.max(getHeight(node.left), getHeight(node.right)) + 1;rightChild.balanceFactor = Math.max(getHeight(rightChild.left), getHeight(rightChild.right)) + 1;return rightChild;}// 右旋转private AVLNode rightRotate(AVLNode node) {AVLNode leftChild = node.left;AVLNode rightGrandChild = leftChild.right;leftChild.right = node;node.left = rightGrandChild;node.balanceFactor = Math.max(getHeight(node.left), getHeight(node.right)) + 1;leftChild.balanceFactor = Math.max(getHeight(leftChild.left), getHeight(leftChild.right)) + 1;return leftChild;}
}
步骤3:实现AVL树的删除操作
AVL树的删除操作也是对二叉搜索树的删除操作进行了平衡调整。具体步骤如下:
- 在二叉搜索树中删除目标节点。
- 更新删除路径上各节点的平衡因子。
- 如果某个节点的平衡因子绝对值大于1,则进行相应的平衡调整。
class AVLTree {// ...// 删除节点public void delete(int value) {root = deleteNode(root, value);}// 删除节点辅助函数private AVLNode deleteNode(AVLNode node, int value) {if (node == null) {return null;}if (value < node.value) {node.left = deleteNode(node.left, value);} else if (value > node.value) {node.right = deleteNode(node.right, value);} else {if (node.left == null || node.right == null) {node = (node.left != null) ? node.left :node.right;} else {AVLNode minNode = findMin(node.right);node.value = minNode.value;node.right = deleteNode(node.right, minNode.value);}}if (node == null) {return null;}// 更新平衡因子node.balanceFactor = Math.max(getHeight(node.left), getHeight(node.right)) + 1;// 平衡调整int balance = getBalanceFactor(node);if (balance > 1 && getBalanceFactor(node.left) >= 0) {return rightRotate(node);}if (balance > 1 && getBalanceFactor(node.left) < 0) {node.left = leftRotate(node.left);return rightRotate(node);}if (balance < -1 && getBalanceFactor(node.right) <= 0) {return leftRotate(node);}if (balance < -1 && getBalanceFactor(node.right) > 0) {node.right = rightRotate(node.right);return leftRotate(node);}return node;}// 查找最小节点private AVLNode findMin(AVLNode node) {while (node.left != null) {node = node.left;}return node;}
}
步骤4:实现AVL树的查找操作
AVL树的查找操作与二叉搜索树的查找操作相同,不需要进行平衡调整。
class AVLTree {// ...// 查找节点public boolean search(int value) {return searchNode(root, value);}// 查找节点辅助函数private boolean searchNode(AVLNode node, int value) {if (node == null) {return false;}if (value < node.value) {return searchNode(node.left, value);} else if (value > node.value) {return searchNode(node.right, value);} else {return true;}}
}
步骤5:测试AVL树的各个操作
public class Main {public static void main(String[] args) {AVLTree tree = new AVLTree();tree.insert(10);tree.insert(20);tree.insert(30);tree.insert(40);tree.insert(50);tree.insert(25);System.out.println("AVL树是否包含值30:" + tree.search(30)); // 输出:truetree.delete(30);System.out.println("AVL树是否包含值30:" + tree.search(30)); // 输出:false}
}
相关文章:
Java手写AVL树
Java手写AVL树 1. AVL树实现思路原理 为了解释AVL树的实现思路原理,下面使用Mermanid代码表示该算法的思维导图: #mermaid-svg-ycH8kKpzVk2HWEby {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid…...
运维自动化:提高效率的秘诀
🌷🍁 博主猫头虎(🐅🐾)带您 Go to New World✨🍁 🦄 博客首页——🐅🐾猫头虎的博客🎐 🐳 《面试题大全专栏》 🦕 文章图文…...
C++设计模式_05_Observer 观察者模式
接上篇,本篇将会介绍C设计模式中的Observer 观察者模式,和前2篇模板方法Template Method及Strategy 策略模式一样,仍属于“组件协作”模式。Observer 在某些领域也叫做 Event 。 文章目录 1. 动机( Motivation)2. 代码…...
github网站打不开,hosts文件配置
首先获取github官网的ip地址, 打开cmd,输入ping github.com 配置: #github 140.82.114.4 github.com 199.232.69.194 github.global.ssl.fastly.net 185.199.108.153 assets-cdn.github.com 185.199.110.153 assets-cdn.github.com 185.199…...
总结PCB设计的经验
一般PCB基本设计流程如下:前期准备->PCB结构设计->PCB布局->布线->布线优化和丝印->网络和DRC检查和结构检查->制版。: : 第一:前期准备。这包括准备元件库和原理图。“工欲善其事,必先利其器”,要做出一…...
HCIE-HCS规划设计搭建
1、相关术语 1、等价路由 等价路由(Equal-cost routing)是一种网络路由策略,用于在网络中选择多个具有相同路由度量(路由距离或成本)的最佳路径之一来转发数据流量。 当存在多个路径具有相同的路由度量时,…...
c语言输出杨辉三角
#include<stdio.h> int main() {int x 0; //表示杨辉三角的的大小int y 1;printf("请输入x的值: ");scanf("%d", &x);for (int i 0; i < x; i) {for (int j 0; j < i; j) {if (j 0 || i 0) {y 1;}else {y y * (i - j 1) / j;}pri…...
性能测试-持续测试及性能测试建设(22)
什么是持续测试? 持续测试定义为:在软件交付流水线中执行自动化测试的过程,目的是获得关于预发布软件业务风险的即时反馈。 完成持续测试,我们还是需要回到定义中,它有3个关键词:软件交付流水线、自动化测试、即时反馈。 首先,持续测试需要具备一条完整的流水线,其代表…...
嵌入式C 语言中的三块技术难点
C 语言在嵌入式学习中是必备的知识,甚至大部分操作系统都要围绕 C 语言进行,而其中有三块技术难点,几乎是公认级别的“难啃的硬骨头”。 今天就来带你将这三块硬骨头细细拆解开来,一定让你看明白了。 0x01 指针 指针是公认…...
【斗破年番】紫研新形象,萧炎终成翻海印,救援月媚,三宗决战
Hello,小伙伴们,我是小郑继续为大家深度解析斗破年番。 斗破苍穹年番动画更新了,小医仙帅气回归,萧炎紫妍成功进入山谷闭关苦修,美杜莎女王守护没多久,就因蛇人族求救离开。从官方公布的最新预告来看,萧炎紫…...
差分方程模型:国民总收入(GDP)的乘数-加速数模型
【背景知识-凯恩斯经济增长模型】 凯恩斯(John M.Keynes)建立了著名的国民经济增长模型。令Y表示国民总收入,C表示总消费,E为总支出,I表示投资,G为政府的投入(如基建等)。那么有 【6.1】 其中࿰…...
【C语言】指针和数组笔试题解析(1)
指针是C语言的灵魂,他的玩法多种多样,这篇文章带来指针的笔试题详解,可以帮助我们更好的理解与巩固指针的知识 目录 预备知识:题目:一维数组:二维数组: 题目比较多,但切记戒骄戒躁&a…...
Vue中组件的三种注册方式
组件的注册 1.全局注册: 在全局注册中,你需要确保在 Vue 根实例之前导入并注册组件。通常,你会在入口文件(例如 main.js)中执行这些操作。 // main.jsimport Vue from vue; import App from ./App.vue;// 导入全局组…...
docker 和k8s 入门
docker 和k8s 入门 本文是云原生的学习记录,可以参考以下文档 k8s https://www.yuque.com/leifengyang/oncloud 相关视频教程可参考如下 https://www.bilibili.com/video/BV13Q4y1C7hS?p2&vd_source0882f549dac54045384d4a921596e234 相对于公有云&#x…...
基于Yolov8的交通标志牌(TT100K)识别检测系统
1.Yolov8介绍 Ultralytics YOLOv8是Ultralytics公司开发的YOLO目标检测和图像分割模型的最新版本。YOLOv8是一种尖端的、最先进的(SOTA)模型,它建立在先前YOLO成功基础上,并引入了新功能和改进,以进一步提升性能和灵活…...
使用Python编写一个多线程的12306抢票程序
国庆长假即将到来,大家纷纷计划着自己的旅行行程。然而,对于很多人来说,抢购火车票人们成了一个令人头疼的问题。12306网站的服务器经常因为流量高而崩溃,导致抢票变得越来越严重异常困难。 首先,让我们来了解一下1230…...
DT Paint Effects工具(三)
管 分支 使用细枝 叶 力 使用湍流 流动画 渲染全局参数 建造盆栽植物...
SpringBoot整合Mybatis
目录 (1)引入依赖 (2)编写Mapper接口 (3)编写Mapper映射文件 (4)编写yml配置文件 (5)编写测试类 (1)引入依赖 <dependency>…...
Java后端使用POST请求向mysql中插入Json数据的问题
1.后端请求正常 但数据表中value没有值 原因 json数据属性不符合spring解析格式,json属性名称的大写字母不符合spring要求 以下为为错误示范 1 Test 以大写字母开头, 2 tTest 小写字母开头,但是第二个字母是大写解决方案 实体类属性加上Jso…...
豆瓣图书评分数据的可视化分析
导语 豆瓣是一个提供图书、电影、音乐等文化产品的社区平台,用户可以在上面发表自己的评价和评论,形成一个丰富的文化数据库。本文将介绍如何使用爬虫技术获取豆瓣图书的评分数据,并进行可视化分析,探索不同类型、不同年代、不同…...
铭豹扩展坞 USB转网口 突然无法识别解决方法
当 USB 转网口扩展坞在一台笔记本上无法识别,但在其他电脑上正常工作时,问题通常出在笔记本自身或其与扩展坞的兼容性上。以下是系统化的定位思路和排查步骤,帮助你快速找到故障原因: 背景: 一个M-pard(铭豹)扩展坞的网卡突然无法识别了,扩展出来的三个USB接口正常。…...
大数据学习栈记——Neo4j的安装与使用
本文介绍图数据库Neofj的安装与使用,操作系统:Ubuntu24.04,Neofj版本:2025.04.0。 Apt安装 Neofj可以进行官网安装:Neo4j Deployment Center - Graph Database & Analytics 我这里安装是添加软件源的方法 最新版…...
以下是对华为 HarmonyOS NETX 5属性动画(ArkTS)文档的结构化整理,通过层级标题、表格和代码块提升可读性:
一、属性动画概述NETX 作用:实现组件通用属性的渐变过渡效果,提升用户体验。支持属性:width、height、backgroundColor、opacity、scale、rotate、translate等。注意事项: 布局类属性(如宽高)变化时&#…...
安宝特方案丨XRSOP人员作业标准化管理平台:AR智慧点检验收套件
在选煤厂、化工厂、钢铁厂等过程生产型企业,其生产设备的运行效率和非计划停机对工业制造效益有较大影响。 随着企业自动化和智能化建设的推进,需提前预防假检、错检、漏检,推动智慧生产运维系统数据的流动和现场赋能应用。同时,…...
cf2117E
原题链接:https://codeforces.com/contest/2117/problem/E 题目背景: 给定两个数组a,b,可以执行多次以下操作:选择 i (1 < i < n - 1),并设置 或,也可以在执行上述操作前执行一次删除任意 和 。求…...
MODBUS TCP转CANopen 技术赋能高效协同作业
在现代工业自动化领域,MODBUS TCP和CANopen两种通讯协议因其稳定性和高效性被广泛应用于各种设备和系统中。而随着科技的不断进步,这两种通讯协议也正在被逐步融合,形成了一种新型的通讯方式——开疆智能MODBUS TCP转CANopen网关KJ-TCPC-CANP…...
力扣-35.搜索插入位置
题目描述 给定一个排序数组和一个目标值,在数组中找到目标值,并返回其索引。如果目标值不存在于数组中,返回它将会被按顺序插入的位置。 请必须使用时间复杂度为 O(log n) 的算法。 class Solution {public int searchInsert(int[] nums, …...
【从零学习JVM|第三篇】类的生命周期(高频面试题)
前言: 在Java编程中,类的生命周期是指类从被加载到内存中开始,到被卸载出内存为止的整个过程。了解类的生命周期对于理解Java程序的运行机制以及性能优化非常重要。本文会深入探寻类的生命周期,让读者对此有深刻印象。 目录 …...
【JVM面试篇】高频八股汇总——类加载和类加载器
目录 1. 讲一下类加载过程? 2. Java创建对象的过程? 3. 对象的生命周期? 4. 类加载器有哪些? 5. 双亲委派模型的作用(好处)? 6. 讲一下类的加载和双亲委派原则? 7. 双亲委派模…...
图解JavaScript原型:原型链及其分析 | JavaScript图解
忽略该图的细节(如内存地址值没有用二进制) 以下是对该图进一步的理解和总结 1. JS 对象概念的辨析 对象是什么:保存在堆中一块区域,同时在栈中有一块区域保存其在堆中的地址(也就是我们通常说的该变量指向谁&…...
