符合直觉的完美平衡树?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

相关文章

windos server2022里的DFS配置的实现

《windosserver2022里的DFS配置的实现》DFS是WindowsServer操作系统提供的一种功能,用于在多台服务器上集中管理共享文件夹和文件的分布式存储解决方案,本文就来介绍一下wi... 目录什么是DFS?优势:应用场景:DFS配置步骤什么是DFS?DFS指的是分布式文件系统(Distr

NFS实现多服务器文件的共享的方法步骤

《NFS实现多服务器文件的共享的方法步骤》NFS允许网络中的计算机之间共享资源,客户端可以透明地读写远端NFS服务器上的文件,本文就来介绍一下NFS实现多服务器文件的共享的方法步骤,感兴趣的可以了解一... 目录一、简介二、部署1、准备1、服务端和客户端:安装nfs-utils2、服务端:创建共享目录3、服

C#使用yield关键字实现提升迭代性能与效率

《C#使用yield关键字实现提升迭代性能与效率》yield关键字在C#中简化了数据迭代的方式,实现了按需生成数据,自动维护迭代状态,本文主要来聊聊如何使用yield关键字实现提升迭代性能与效率,感兴... 目录前言传统迭代和yield迭代方式对比yield延迟加载按需获取数据yield break显式示迭

Python实现高效地读写大型文件

《Python实现高效地读写大型文件》Python如何读写的是大型文件,有没有什么方法来提高效率呢,这篇文章就来和大家聊聊如何在Python中高效地读写大型文件,需要的可以了解下... 目录一、逐行读取大型文件二、分块读取大型文件三、使用 mmap 模块进行内存映射文件操作(适用于大文件)四、使用 pand

python实现pdf转word和excel的示例代码

《python实现pdf转word和excel的示例代码》本文主要介绍了python实现pdf转word和excel的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价... 目录一、引言二、python编程1,PDF转Word2,PDF转Excel三、前端页面效果展示总结一

在MyBatis的XML映射文件中<trim>元素所有场景下的完整使用示例代码

《在MyBatis的XML映射文件中<trim>元素所有场景下的完整使用示例代码》在MyBatis的XML映射文件中,trim元素用于动态添加SQL语句的一部分,处理前缀、后缀及多余的逗号或连接符,示... 在MyBATis的XML映射文件中,<trim>元素用于动态地添加SQL语句的一部分,例如SET或W

Python xmltodict实现简化XML数据处理

《Pythonxmltodict实现简化XML数据处理》Python社区为提供了xmltodict库,它专为简化XML与Python数据结构的转换而设计,本文主要来为大家介绍一下如何使用xmltod... 目录一、引言二、XMLtodict介绍设计理念适用场景三、功能参数与属性1、parse函数2、unpa

C#实现获得某个枚举的所有名称

《C#实现获得某个枚举的所有名称》这篇文章主要为大家详细介绍了C#如何实现获得某个枚举的所有名称,文中的示例代码讲解详细,具有一定的借鉴价值,有需要的小伙伴可以参考一下... C#中获得某个枚举的所有名称using System;using System.Collections.Generic;usi

Go语言实现将中文转化为拼音功能

《Go语言实现将中文转化为拼音功能》这篇文章主要为大家详细介绍了Go语言中如何实现将中文转化为拼音功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 有这么一个需求:新用户入职 创建一系列账号比较麻烦,打算通过接口传入姓名进行初始化。想把姓名转化成拼音。因为有些账号即需要中文也需要英

C# 读写ini文件操作实现

《C#读写ini文件操作实现》本文主要介绍了C#读写ini文件操作实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录一、INI文件结构二、读取INI文件中的数据在C#应用程序中,常将INI文件作为配置文件,用于存储应用程序的