符合直觉的完美平衡树?AVL树(自平衡二叉查找树)的Lua实现与测试场景实例

本文主要是介绍符合直觉的完美平衡树?AVL树(自平衡二叉查找树)的Lua实现与测试场景实例,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

AVL树(Adelson-Velsky and Landis Tree)是计算机科学中最早被发明的自平衡二叉查找树。在AVL树中,任一节点对应的两棵子树的最大高度差为1,因此它也被称为高度平衡树。查找、插入和删除在平均和最坏情况下的时间复杂度都是O(log {n})。增加和删除元素的操作则可能需要借由一次或多次树旋转,以实现树的重新平衡。

文章目录

  • 一、AVL树是什么?如何维持自身平衡性?
  • 二、AVL树中的节点旋转概念:
  • 三、代码实现部分:
    • 1.Node节点类
    • 2.AVLTree类
  • 测试场景实例


一、AVL树是什么?如何维持自身平衡性?

AVL树本质上是一颗二叉查找树,但是它又具有以下特点:它是一棵空树或它的左右两个子树的高度差的绝对值不超过1,并且左右两个子树都是一棵平衡二叉树。 在AVL树中任何节点的两个子树的高度最大差别为一,所以它也被称为平衡二叉树。
AVL树在插入节点的过程中如何保证树的平衡性(并不是每次调整所有的节点),在插入时检查树的叶子节点,找到第一个失去平衡的节点(最小失衡子树), 调整该节点(子树),整棵树就达到平衡了。

二、AVL树中的节点旋转概念:

AVL树维持平衡的四种Case:LL, RR, LR, RL (子节点-父节点-祖父节点) 三个节点的关系可以通过自己绘图来整理
LL (Left Node of Left Child 左孩子的左节点) 右旋
RR (Right Node of Right Child 右孩子的右节点) 左旋
LR (Left Node of Right Child 右孩子的左节点) 先左旋再右旋
RL (Right Node of Left Child 左孩子的右节点) 先右旋再左旋

三、代码实现部分:

1.Node节点类

代码如下(示例):

-- Node class
local Node = {}
Node.__index = Nodefunction Node.new(key, value)-- create a new node with key and valuelocal self = setmetatable({}, Node)self.key = keyself.value = valueself.left = nilself.right = nilself.height = 1return self
endfunction Node:get_balance()-- get the balance factor of the nodelocal left_height = self.left and self.left.height or 0local right_height = self.right and self.right.height or 0return left_height - right_height
endfunction Node:update_height()-- update the height of the nodelocal left_height = self.left and self.left.height or 0local right_height = self.right and self.right.height or 0self.height = 1 + math.max(left_height, right_height)
endfunction Node:rotate_right()-- rotate the node to the rightlocal y = self.leftself.left = y.righty.right = selfself:update_height()y:update_height()return y
endfunction Node:rotate_left()-- rotate the node to the leftlocal y = self.rightself.right = y.lefty.left = selfself:update_height()y:update_height()return y
endfunction Node:rebalance()-- rebalance the node if necessaryself:update_height()local balance = self:get_balance()if balance > 1 thenif self.left:get_balance() < 0 thenself.left = self.left:rotate_left()endreturn self:rotate_right()elseif balance < -1 thenif self.right:get_balance() > 0 thenself.right = self.right:rotate_right()endreturn self:rotate_left()elsereturn selfend
end

2.AVLTree类

代码如下(示例):

-- AVLTree class
local AVLTree = {}
AVLTree.__index = AVLTreefunction AVLTree:new()-- create a new AVL treelocal self = setmetatable({}, AVLTree)self.root = nilreturn self
endfunction AVLTree:insert(key, value)-- insert a key-value pair into the treeself.root = self:_insert(self.root, key, value)
endfunction AVLTree:_insert(node, key, value)-- recursive insert functionif not node thenreturn Node.new(key, value)elseif key < node.key thennode.left = self:_insert(node.left, key, value)elseif key > node.key thennode.right = self:_insert(node.right, key, value)elsenode.value = valueendreturn node:rebalance()
endfunction AVLTree:remove(key)-- remove a key from the treeself.root = self:_remove(self.root, key)
endfunction AVLTree:_remove(node, key)-- recursive remove functionif not node thenreturn nilelseif key < node.key thennode.left = self:_remove(node.left, key)elseif key > node.key thennode.right = self:_remove(node.right, key)elseif not node.left and not node.right thenreturn nilelseif not node.left thenreturn node.rightelseif not node.right thenreturn node.leftelselocal successor = self:_find_min(node.right)node.key = successor.keynode.value = successor.valuenode.right = self:_remove(node.right, successor.key)endendreturn node:rebalance()
endfunction AVLTree:find(key)-- find the value for a given key in the treelocal node = self:_find(self.root, key)return node and node.value or nil
endfunction AVLTree:_find(node, key)-- recursive find functionif not node thenreturn nilelseif key < node.key thenreturn self:_find(node.left, key)elseif key > node.key thenreturn self:_find(node.right, key)elsereturn nodeend
endfunction AVLTree:_find_min(node)-- find the minimum key in the treewhile node.left donode = node.leftendreturn node
endfunction AVLTree:traverse(visit)-- traverse the tree in-order and apply the visit function to each nodeself:_traverse(self.root, visit)
endfunction AVLTree:_traverse(node, visit)-- recursive in-order traversalif node thenself:_traverse(node.left, visit)visit(node.key, node.value)self:_traverse(node.right, visit)end
endreturn AVLTree

测试场景实例

按照惯例测试场景实例代码不放出,我是在Unity环境中建立的Canvas上绘制的节点图。期间遇到了一些小麻烦,本应该是一颗"完美的平衡二叉树",绘制起来却显得有些丑陋,尤其是树的高度越大,整棵树底部越显得"臃肿"…, 在Canvas中的排版下看起来并不体面。

不过整棵树建立的结构还是很正确的,下图所示:
测试场景实例图

这篇关于符合直觉的完美平衡树?AVL树(自平衡二叉查找树)的Lua实现与测试场景实例的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

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

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

Python的Darts库实现时间序列预测

《Python的Darts库实现时间序列预测》Darts一个集统计、机器学习与深度学习模型于一体的Python时间序列预测库,本文主要介绍了Python的Darts库实现时间序列预测,感兴趣的可以了解... 目录目录一、什么是 Darts?二、安装与基本配置安装 Darts导入基础模块三、时间序列数据结构与

Python使用FastAPI实现大文件分片上传与断点续传功能

《Python使用FastAPI实现大文件分片上传与断点续传功能》大文件直传常遇到超时、网络抖动失败、失败后只能重传的问题,分片上传+断点续传可以把大文件拆成若干小块逐个上传,并在中断后从已完成分片继... 目录一、接口设计二、服务端实现(FastAPI)2.1 运行环境2.2 目录结构建议2.3 serv

C#实现千万数据秒级导入的代码

《C#实现千万数据秒级导入的代码》在实际开发中excel导入很常见,现代社会中很容易遇到大数据处理业务,所以本文我就给大家分享一下千万数据秒级导入怎么实现,文中有详细的代码示例供大家参考,需要的朋友可... 目录前言一、数据存储二、处理逻辑优化前代码处理逻辑优化后的代码总结前言在实际开发中excel导入很

SpringBoot+RustFS 实现文件切片极速上传的实例代码

《SpringBoot+RustFS实现文件切片极速上传的实例代码》本文介绍利用SpringBoot和RustFS构建高性能文件切片上传系统,实现大文件秒传、断点续传和分片上传等功能,具有一定的参考... 目录一、为什么选择 RustFS + SpringBoot?二、环境准备与部署2.1 安装 RustF

Nginx部署HTTP/3的实现步骤

《Nginx部署HTTP/3的实现步骤》本文介绍了在Nginx中部署HTTP/3的详细步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录前提条件第一步:安装必要的依赖库第二步:获取并构建 BoringSSL第三步:获取 Nginx

MyBatis Plus实现时间字段自动填充的完整方案

《MyBatisPlus实现时间字段自动填充的完整方案》在日常开发中,我们经常需要记录数据的创建时间和更新时间,传统的做法是在每次插入或更新操作时手动设置这些时间字段,这种方式不仅繁琐,还容易遗漏,... 目录前言解决目标技术栈实现步骤1. 实体类注解配置2. 创建元数据处理器3. 服务层代码优化填充机制详

Python实现Excel批量样式修改器(附完整代码)

《Python实现Excel批量样式修改器(附完整代码)》这篇文章主要为大家详细介绍了如何使用Python实现一个Excel批量样式修改器,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一... 目录前言功能特性核心功能界面特性系统要求安装说明使用指南基本操作流程高级功能技术实现核心技术栈关键函

Java实现字节字符转bcd编码

《Java实现字节字符转bcd编码》BCD是一种将十进制数字编码为二进制的表示方式,常用于数字显示和存储,本文将介绍如何在Java中实现字节字符转BCD码的过程,需要的小伙伴可以了解下... 目录前言BCD码是什么Java实现字节转bcd编码方法补充总结前言BCD码(Binary-Coded Decima

防止Linux rm命令误操作的多场景防护方案与实践

《防止Linuxrm命令误操作的多场景防护方案与实践》在Linux系统中,rm命令是删除文件和目录的高效工具,但一旦误操作,如执行rm-rf/或rm-rf/*,极易导致系统数据灾难,本文针对不同场景... 目录引言理解 rm 命令及误操作风险rm 命令基础常见误操作案例防护方案使用 rm编程 别名及安全删除