返回顶部
首页 > 资讯 > 后端开发 > Python >详细理解平衡二叉树AVL与Python实
  • 631
分享到

详细理解平衡二叉树AVL与Python实

二叉树详细Python 2023-01-30 23:01:34 631人浏览 八月长安

Python 官方文档:入门教程 => 点击学习

摘要

上一篇文章讨论的二叉搜索树,其时间复杂度最好的情况下是O(log(n)),但是最坏的情况是O(n),什么时候是O(n)呢? 像这样: 如果先插入10,再插入20,再插入30,再插入40就会成上边这个样子 这个就像是双向链表,我们期望它

上一篇文章讨论的二叉搜索树,其时间复杂度最好的情况下是O(log(n)),但是最坏的情况是O(n),什么时候是O(n)呢?

像这样:

如果先插入10,再插入20,再插入30,再插入40就会成上边这个样子

这个就像是双向链表,我们期望它是下面这个样子:

所以我们希望有一种策略能够将第一个图变成第二个图,或者说使树的结构不会产生像第一种图的形式

实现这种策略的一种方式是AVL树

AVL树的名称是以它的发明家的名字命名的:Adel’son-Vel’skii和Landis

满足高度平衡属性的二叉树就是AVL树

高度平衡属性是:对于树中的每一个位置p,p的孩子的高度最多相差1

很显然前言中的第一个图并不满足高度平衡属性,第二个是满足的。

同时高度平衡属性也意味着一颗AVL树的子树同样是AVL树

并且可以通过证明(这里就不再证了)得到AVL树的高度是O(log n)

所以得出结论,AVL树可以使时间复杂度保持O(log n)

接下来的问题就是怎样保持二叉树的高度平衡属性

要保持高度平衡属性的原因是破坏了高度平衡属性

破坏的方式有两种:添加节点与删除节点

添加节点如图:

添加50的时候,就会破坏高度平衡属性

删除节点如图:

删除10的时候也会破坏高度平衡属性

最后,不论是添加节点还是删除节点,都会使树变成非高度平衡的状态,这种非高度平衡的状态有4种:

1.LL

LL是left-left,可以理解为:首先它不平衡,其次根节点的左子树比右子树高,并且根节点的左子树的左子树比根节点的左子树的右子树高。(从上到下都是左边高)

2.LR

LR是left-right,可以理解为:首先它不平衡,其次根节点的左子树比右子树高,并且根节点的左子树的右子树比根节点的左子树的左子树高。(从上到下先左高后右高)

3.RR

RR是right-right,可以理解为:首先它不平衡,其次根节点的右子树比左子树高,并且根节点的右子树的右子树比根节点的右子树的左子树高。(从上到下都是右边高)

4.RL

RL是right-left,可以理解为:首先它不平衡,其次根节点的右子树比左子树高,并且根节点的右子树的左子树比根节点的右子树的右子树高。(从上到下先右高后左高)

最后,判断是哪种形式的非平衡状态,一定要从不平衡的节点位置看,并不是看4层,比如:

这里只有3层节点,不平衡的节点是20,20的左子树比右子树高,10的左子树比右子树高,所以是LL。(这里的高定义为节点5的高度为1,空节点的高度为0)

接下来是保持高度平衡的调整策略:

同样对于4种不同的形式有4种解决方案:

1.LL

这个变换就像是以10为中心,向右旋转,使10变成根节点,10的左子树不变,右子树变成了20,多余出的15正好挂在由于变换失去了左子树的20的左边。变换后结点从左到右的顺序依然没有变,所以15是正好挂在20的左边的。

2.RR

RR与LL形式差不多,只不顾是反着来的。相当于进行一次左旋转。

RR与LL都只进行一次旋转即可,而LR与RL需要进行两次旋转

3.LR

第一次相当于对5、10、15、17这棵子树进行了一次RR旋转,旋转方式与之前的RR方式相同,就像是以15为中心向左旋转,旋转的结果使得整棵树变成了LL的不平衡形态,然后再按照LL的旋转方式对整棵树处理。

4.RL

RL同样是LR的相反模式,先将22、25、30、40这棵子树进行LL旋转,再将整棵树进行RR旋转

理解了avl保持平衡从方式后,就可以用代码来实现了

我们使用AVL对上一篇文章中的有序映射进行优化

因为AVL依赖于节点的高度,所以首先要重写一下node类:


class AvlTree(OrderedMap):

    class Node(OrderedMap.Node):
        def __init__(self, element, parent=None, left=None, right=None):
            super().__init__(element,parent,left,right)
            self.height = 0

        def left_height(self):
            return self.left.height if self.left is not None else 0

        def right_height(self):
            return self.right.height if self.right is not None else 0

接下来定义4中调整的非公开方法


def _left_left(self,p):
    this = p.node        # 有变化的就4个节点
    left = this.left
    parent = this.parent
    left_right = this.left.right
    if parent is not None:
        if this is parent.left:
            parent.left = left
        else:
            parent.right = left
    else:
        self._root = left
    this.parent = left
    left.parent = parent
    this.left = left_right
    left.right = this
    if left_right is not None:
        left_right.parent = this

def _right_right(self,p):
    this = p.node                 # 有变化的就4个节点
    right = this.right
    parent = this.parent
    right_left = this.right.left
    if parent is not None:
        if this is parent.left:
            parent.left = right
        else:
            parent.right = right
    else:
        self._root = right
    this.parent = right
    right.parent = parent
    this.right = right_left
    right.left = this
    if right_left is not None:
        right_left.parent = this

def _left_right(self,p):
    self._right_right(self.left(p))
    self._left_left(p)

def _right_left(self,p):
    self._left_left(self.right(p))
    self._right_right(p)

然后是用于平衡二叉树的方法,也就是根据情况调用上边那4种策略


def _isbalanced(self,p):
    """判断节点是否平衡"""

    return abs(p.node.left_height() - p.node.right_height()) <= 1

def _recompute_height(self,p):
    """重新计算高度"""
    p.node.height = 1 + max(p.node.left_height(),p.node.right_height())

def _rebalanced(self,p):
    while p is not None:
        if self._isbalanced(p):
            self._recompute_height(p)
            p = self.parent(p)
        else:

            if p.node.left_height()>p.node.right_height() and p.node.left.left_height()>p.node.left.right_height():
                # LL的情况,只有自己和左孩子的高度可能变化
                self._left_left(p)
            elif p.node.right_height()>p.node.left_height() and p.node.right.right_height()>p.node.right.left_height():
                # RR的情况,只有自己和右孩子的高度可能变化
                self._right_right(p)
            elif p.node.left_height()>p.node.right_height() and p.node.left.left_height()<p.node.left.right_height():
                # LR的情况,只有自己和左孩子和左孩子的右孩子的高度可能变化
                left = self.left(p)
                self._left_right(p)
                self._recompute_height(left)
            else:
                # RL的情况,只有自己和右孩子和右孩子的左孩子的高度可能变化
                right = self.right(p)
                self._right_left(p)
                self._recompute_height(right)
            while p is not None:
                # 调整所有p的祖先的高度
                self._recompute_height(p)
                p = self.parent(p)

然后把方法封装成删除时和插入时的两个方法,虽然执行的内容是相同的


def _rebalanced_insert(self,p):
    """插入时的平衡调整"""
    self._rebalanced(p)

def _rebalanced_delete(self, p):
    """删除时的平衡调整"""
    self._rebalanced(p)

最后重写一下setitem方法与删除时调用的方法


def __setitem__(self, k, v):
    """优化setitem"""
    if self.is_empty():
        leaf = self.add_root(self._Item(k, v))
    else:

        p = self._subtree_search(self.root(), k)
        if p.key() == k:
            p.element().value = v
            return
        else:
            item = self._Item(k, v)
            if p.key() < k:
                leaf = self.add_right(p, item)
            else:
                leaf = self.add_left(p, item)
    self._rebalanced_insert(leaf)

def mapdelete(self, p):
    if self.left(p) and self.right(p):  # 两个孩子都有的时候
        replacement = self._subtree_last_position(
            self.left(p))  # 用左子树最右位置代替
        self.replace(p, replacement.element())
        p = replacement
    parent = self.parent(p)
    self.delete(p)
    self._rebalanced_delete(parent)

在实现4种平衡策略时,一定要记着将整棵树的根节点更新,不然遍历的时候,根节点指的就不是真正的根节点了。

--结束END--

本文标题: 详细理解平衡二叉树AVL与Python实

本文链接: https://lsjlt.com/news/181133.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • 详细理解平衡二叉树AVL与Python实
    上一篇文章讨论的二叉搜索树,其时间复杂度最好的情况下是O(log(n)),但是最坏的情况是O(n),什么时候是O(n)呢? 像这样: 如果先插入10,再插入20,再插入30,再插入40就会成上边这个样子 这个就像是双向链表,我们期望它...
    99+
    2023-01-30
    二叉树 详细 Python
  • 什么是平衡二叉树AVL
    什么是平衡二叉树AVL,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。前言Wiki:在计算机科学中,AVL树是最早被发明的自平衡二叉查找树。在AVL树中,任一节点...
    99+
    2023-06-04
  • C语言平衡二叉树详解
    目录调整措施:一、单旋转二、双旋转AVL树的删除操作:删除分为以下几种情况:1.要删除的节点是当前根节点T。2、要删除的节点元素值小于当前根节点T值,在左子树中进行删除。3、要删除的...
    99+
    2024-04-02
  • Python实现二叉排序树与平衡二叉树的示例代码
    目录前言1. 二叉排序树1.1 构建一棵二叉排序树1.2 二叉排序树的数据结构1.3 实现二叉排序树类中的方法:2. 平衡二叉排序树2.1 二叉平衡排序树的数据结构3. 总结前言 什...
    99+
    2024-04-02
  • Python树表查找(二叉排序树、平衡二叉树)
    目录什么是树表查询?1. 二叉排序树1.1 构建一棵二叉排序树1.2 二叉排序树的数据结构1.3 实现二叉排序树类中的方法:3. 平衡二叉排序树3.1 二叉平衡排序树的数据结构4. ...
    99+
    2023-01-07
    python建立二叉排序树 二叉排序树python实现 python实现平衡二叉树遍历
  • Java实现红黑树(平衡二叉树)的详细过程
    目录前言红黑二叉查找树2-3树2-3树的插入操作实现红黑二叉树结尾前言 在实现红黑树之前,我们先来了解一下符号表。 符号表的描述借鉴了Algorithms第四版,详情在:https...
    99+
    2024-04-02
  • C语言之平衡二叉树详解
    目录什么是平衡二叉树平衡二叉树的基本特点为什么会出现平衡二叉树二叉树四种不平衡的情况C语言实现平衡二叉树什么是平衡二叉树 平衡二叉树是具有平衡属性的有序二叉树,所谓的平衡即当前树的左...
    99+
    2023-05-17
    C语言二叉树 C语言平衡二叉树
  • 详解如何用c++实现平衡二叉树
    目录一、概述二、平衡二叉树不平衡的情形三、调整措施3.1、单旋转3.2、双旋转四、AVL树的删除操作五、代码实现一、概述 平衡二叉树具有以下性质:它是一 棵空树或它的左右两个子树的高...
    99+
    2024-04-02
  • C++超详细讲解树与二叉树
    目录树树的定义树的名词解释树的表示树的存储结构二叉树的概念及结构二叉树的概念二叉树的性质二叉树的存储结构顺序存储结构链式存储结构树 树的定义 Q:什么是树 A:树是一种 非线性 的数...
    99+
    2024-04-02
  • C++实现LeetCode(110.平衡二叉树)
    [LeetCode] 110.Balanced Binary Tree 平衡二叉树 Given a binary tree, determine if it is height-ba...
    99+
    2024-04-02
  • C++怎么实现平衡二叉树
    本篇内容介绍了“C++怎么实现平衡二叉树”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!平衡二叉树Given a binary tree, d...
    99+
    2023-06-20
  • Java如何实现平衡二叉树
    小编给大家分享一下Java如何实现平衡二叉树,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!1 平衡二叉树的概述为了避免极端情况下二叉搜索树退化为链表,影响查找效率...
    99+
    2023-06-28
  • Java平衡二叉树怎么实现
    本篇内容主要讲解“Java平衡二叉树怎么实现”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Java平衡二叉树怎么实现”吧!什么是二叉搜索树简单来说,就是方便搜索的二叉树,是一种具备特定结构的二叉...
    99+
    2023-06-29
  • Java平衡二叉树实例分析
    这篇文章主要讲解了“Java平衡二叉树实例分析”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Java平衡二叉树实例分析”吧!AVL树的引入搜索二叉树有着极高的搜索效率,但是搜索二叉树会出现以...
    99+
    2023-06-30
  • 详解Java数据结构之平衡二叉树
    目录什么是二叉搜索树平衡二叉搜索树平衡二叉搜索树建树程序计算每个节点的高度计算每个节点的平衡因子合并二叉树旋转调整函数整体代码什么是二叉搜索树 简单来说,就是方便搜索的二叉树,是一种...
    99+
    2024-04-02
  • Java源码解析之平衡二叉树
    目录一、平衡二叉树的定义二、平衡二叉树的实现原理三、最终结果一、平衡二叉树的定义 平衡二叉树是一种二叉排序树,其中每一个节点的左子树和右子树的高度差至多等于1 。它是一种高度平衡的二...
    99+
    2024-04-02
  • Java数据结构之平衡二叉树的实现详解
    目录定义结点结构查找算法插入算法LL 型RR 型LR 型RL 型插入方法删除算法概述实例分析代码完整代码定义 动机:二叉查找树的操作实践复杂度由树高度决定,所以希望控制树高,左右子...
    99+
    2024-04-02
  • Java数据结构之平衡二叉树的原理与实现
    目录1 平衡二叉树的概述2 平衡二叉树的实现原理2.1 单旋转2.2 双旋转2.3 总结3 平衡二叉树的构建3.1 类架构3.2 查找的方法3.3 检查是否平衡的方法3.4 插入的方...
    99+
    2024-04-02
  • Java深入分析了解平衡二叉树
    目录AVL树的引入基本概念基础设计RR(左旋)LL(右旋)LR(先左旋再右旋)RL(先右旋再左旋)添加节点删除节点AVL树的引入 搜索二叉树有着极高的搜索效率,但是搜索二叉树会出现以...
    99+
    2024-04-02
  • Java中平衡二叉树的原理是什么
    Java中平衡二叉树的原理是什么?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。一、平衡二叉树的定义平衡二叉树是一种二叉排序树,其中每一个节点的左子树和右子树的高...
    99+
    2023-06-15
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作