数据结构(二), AVL平衡二叉树

Posted by zhoujie on April 11, 2021

一、须知须会

  1. 平衡因子: 二叉树的 左子树 - 右子树 = 高度的差值,在平衡树中可能的值(-1 ,0 ,1)

  2. 平衡: 平衡因子 的绝对值小于 2 (下图第一张为平衡树, 第二张为不平衡树)

  • 平衡树且平衡因子==0

在这里插入图片描述

  • 非平衡树且平衡因子==-2

在这里插入图片描述

  1. 树的旋转: 参考维基百科 树的旋转 在这里插入图片描述 在这里插入图片描述

转轴的移动方向来决定它是左旋还是右旋, 本文中称转轴右移为右旋反之则是左旋

  • 右旋: (Q为树的根节点, P为转轴, 转轴最终被右移) 右旋时,
1
2
3
   转轴的右孩子 = 树的根节点;
   根节点的左孩子 = 转轴的右子树根节点;
   树的根节点 = 转轴;
  • 左旋 (P为树的根节点, Q为转轴, 转轴最终被左移) 左旋时,
1
2
3
   转轴的左孩子 = 树的根节点;
   根节点的右孩子 = 转轴的左子树根节点;
   树的根节点 = 转轴;
  • 中序不变 旋转结束后二叉树的中序始终不变, A<P<B<Q<C

二、简介

AVL树(Adelson-Velsky and Landis Tree)得名于它的发明者G. M. Adelson-Velsky和Evgenii Landis,他们在1962年的论文《An algorithm for the organization of information》中公开了这一数据结构。其实AVL就是在BST的基础上增加了一个平衡(Balance)的属性, 为的是稳固算法复杂度到 O(logn), BST算法复杂度受到树结构的影响会游离于 O(logn)~O(n) 之间, 而加入了平衡属性之后则会降低到恒定为 O(logn)

三、基本特征

  • 首先是 BST 的一种( BST 有的它都有)
  • 是平衡二叉树(根节点的平衡因子绝对值不大于1)
  • 左右子树也是平衡二叉树

四、算法描述

插入, 删除之后是否需要平衡这个树? 如何平衡?

高度

为了判断一个树是否平衡? 引入了高度这个概念, 记录从树的最后一层到当前层数的距离 在这里插入图片描述

所以我们的节点(Node)结构定义如下

1
2
3
4
5
6
    public static class Node {
        private int data;
        private int height;
        private Node left;
        private Node right;
    }

旋转

为了解决树的平衡问题引入了树的旋转

在这里插入图片描述

A, B, C, D代表 子节点, 子树 或者 null

  • 左左 左节点的左节点/子树导致的不平衡, 需要右旋
1
2
3
4
5
6
7
8
9
10
    private Node rotateRight(Node root) {

        Node t = root.left;
        root.left = t.right;
        t.right = root;

        root.height = Math.max(height(root.left), height(root.right)) + 1;
        t.height = Math.max(height(t.left), height(t.right)) + 1;
        return t;
    }

右右 右节点的右节点/子树导致的不平衡, 需要左旋

1
2
3
4
5
6
7
8
9
10
    private Node rotateLeft(Node root) {

        Node t = root.right;
        root.right = t.left;
        t.left = root;

        root.height = Math.max(height(root.left), height(root.right)) + 1;
        t.height = Math.max(height(t.left), height(t.right)) + 1;
        return t;
    }

左右 左节点的右节点/子树导致的不平衡, 需要左旋, 右旋

1
2
3
4
    private Node rotateLeftRight(Node root) {
        root.left = rotateLeft(root.left);
        return rotateRight(root);
    }

右左 右节点的左节点/子树导致的不平衡, 需要右旋, 左旋

1
2
3
4
    private Node rotateRightLeft(Node root) {
        root.right = rotateRight(root.right);
        return rotateLeft(root);
    }

查找

和 BST 写法一致

  • 先查找根节点,
  • < 根, 则找左子树;
  • > 根, 则找右子树;
  • = 根, 则找到返回;
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
    public Node search(int num) {
        return doSearch(root, num);
    }

    private Node doSearch(Node root, int num) {
        if (root == null) {
            return null;
        }

        if (root.data == num) {
            return root;
        } else if (root.data > num) {
            return doSearch(root.left, num);
        } else if (root.data < num) {
            return doSearch(root.right, num);
        }
        return null;
    }

算法时间复杂度 对于 n 个节点的树 f(n) = 需要查找的次数 = 二叉树的层数 ~= O(logn) 在这里插入图片描述

插入

  • 比对根节点, 小于就往左节点比对, 大于就往右节点比对
  • 直到需要比对的节点为空, 而这个空就是你需要插入的位置
  • 判断树是否平衡, 否, 则需要判断旋转类型并进行旋转变换
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
    public void insert(int num) {
        root = doInsert(root, num);
    }

    private Node doInsert(Node parent, int num) {
        if (parent == null) {
            parent = new Node(num);
        } else if (num > parent.data) {
            parent.right = doInsert(parent.right, num);

            // 判断树是否失衡
            if (height(parent.right) - height(parent.left) >= 2) {
                // '右右'
                if (num > parent.right.data) {
                    parent = rotateLeft(parent);
                }
                // '右左'
                else {
                    parent = rotateRightLeft(parent);
                }
            }

        } else if (num < parent.data) {
            parent.left = doInsert(parent.left, num);

            // 判断树是否失衡
            if (height(parent.left) - height(parent.right) >= 2) {
                // '左左'
                if (num < parent.left.data) {
                    parent = rotateRight(parent);
                }
                // '左右'
                else {
                    parent = rotateLeftRight(parent);
                }
            }
        }

        // 重新计算旋转之后的高度
        parent.height = Math.max(height(parent.left), height(parent.right)) + 1;

        return parent;
    }

算法时间复杂度: f(n) = 查找插入点的比对次数logn + 一次旋转(单旋或者双旋) + 判断是否平衡 ~= O(logn)

旋转的时间复杂度 O(1) 最多需要单旋或者双旋, 另外, 判断是否平衡的时间复杂度也是 O(1) ( 主要得益于 Node 使用了 height 记录高度, 典型的空间换时间), 这样总得算法复杂度还是 比对的次数 在这里插入图片描述

删除

  • 先查找到目标节点
  • 若: 目标左子树为空, 则, 用目标右子树根节点替换目标
  • 若: 目标右子树为空, 则, 用目标左子树根节点替换目标
  • 若: 都不为空, 则, 选取左子树值最大节点或者右子树最小节点替换目标, 并, 递归删除替换目标的节点
  • 判断树是否平衡, 否, 则需要判断旋转类型并进行旋转变换
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
    public void remove(int num) {
        root = doRemove(root, num);
    }

    private Node doRemove(Node parent, int num) {

        if (parent == null) {
            return null;
        }

        if (num > parent.data) {
            parent.right = doRemove(parent.right, num);

            // 判断树是否平衡
            if (height(parent.left) - height(parent.right) >= 2) {
                Node t = parent.right;
                // `右左`
                if (t == null || height(t.right) < height(t.left)) {
                    parent = rotateRightLeft(parent);
                }
                // `右右`
                else {
                    parent = rotateLeft(parent);
                }
            }

        } else if (num < parent.data) {
            parent.left = doRemove(parent.left, num);

            // 判断树是否平衡
            if (height(parent.right) - height(parent.left) >= 2) {
                Node t = parent.left;
                // `左右`
                if (t == null || height(t.left) < height(t.right)) {
                    parent = rotateLeftRight(parent);
                }
                // `左左`
                else {
                    parent = rotateRight(parent);
                }
            }
        }
        // 找出左子树最大的值或者右子树最小的值替换, 这里选择前者来实现
        else if (parent.left != null && parent.right != null) {

            // 找到左子树最大值替换
            parent.data = findMax(parent.left).data;
            // 删除左子树中用于替换的节点
            parent.left = doRemove(parent.left, parent.data);
        }
        // 左子树为空, 直接用右子树根节点替换被删除的节点
        else if (parent.left == null) {
            parent = parent.right;
        }
        // 右子树为空, 直接用左子树根节点替换被删除的节点
        else if (parent.right == null) {
            parent = parent.left;
        }

        return parent;
    }

    private Node findMax(Node node) {
        if (node == null) {
            return node;
        }
        while (node.right != null) {
            node = node.right;
        }
        return node;
    }

算法复杂度: f(n) = 需要比对的次数 = 查找到目标比对次数logn + 旋转次数logn = O(2logn)

最多需要 logn 次旋转, 来确保平衡, 所以最终的时间复杂度是 O(2logn)

在这里插入图片描述

五、完整代码

就一个AVLTree.java文件搞定, 里面还附有main()函数测试功能, 可直接运行github传送门



-->