从BinarySearchTree到RedBlackTree之AVL

Posted 给个HK.phd读

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了从BinarySearchTree到RedBlackTree之AVL相关的知识,希望对你有一定的参考价值。

AVL树

没什么好多说的,就是可以优化我们在BST中遇到的有序数组形成的"树"退化成链表的情况。
从中引入了旋转、平衡的概念,并不打算解释,单纯练练代码吧。。。。

最后还有一个大头RedBlackTree,不知道鸽多久。。

code

一开始是LeetCode检查平衡的一个题目,挺好的,建议做一做!

// LeetCode 110 判断一棵树是否为平衡树,带领我们初步地认识AVL TREE

/**
 * Definition for a binary tree node.
 * struct TreeNode 
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) 
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) 
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) 
 * ;
 */
/*
class Solution 
public:
    bool isBalanced(TreeNode* root) 
        if(MaxHeight(root) == -1)
            return false;
        return true;
    

    int MaxHeight(TreeNode* root)
        if(root == nullptr)
            return 0; // 代表到了一个空节点,此时默认高度值为0
        
        int LeftHeight = MaxHeight(root->left);    // 返回-1的情况就说明自己的左子树不平衡了
        int RightHeight = MaxHeight(root->right);
        if(LeftHeight == -1 || RightHeight == -1 || abs(LeftHeight - RightHeight) > 1)
            // 需要左子树和右子树均平衡并且以自己为根节点的树也是平衡的
            return -1;
        
        return max(LeftHeight, RightHeight) + 1; // 返回自身的高度
    
;
*/


/*
全部思路按照清华大学->数据结构C语言版实现:
比较精巧减少工程量的内容->把双旋操作进行了分解,省去了一大堆功夫!
switch用得较多,这也是之前一直缺失的[用的较少的语法]。
遇到的bug->switch的case底下不能定义变量!!!否则编译器就炸了
需要注意的point---->  
1. 一定要理解深刻为什么在函数中我们传递了指针的引用!!因为我们要修改那块指针指向的区域而不单单是值!
2. 一定要深刻理解四种旋转来维护平衡的情况,以及画图来计算平衡后的balance_factor。
*/


#include <iostream>
#include <vector>
#include <queue>
using namespace std;

#define LeftHigher 1
#define RightHigher -1
#define Equal 0

template<typename T>
struct AVLNode
    T data;
    int balance_factor;    // 按照清华大学数据结构书的定义 平衡因子为左子树高度减去右子树高度!
    AVLNode<T>* left;
    AVLNode<T>* right;
    AVLNode<T>* parent;
    AVLNode(T n, int bf = 0) : data(n), balance_factor(bf), left(nullptr), right(nullptr), parent(nullptr);
;

template<typename T>
class AVLTREE
private:
    /*
    实际上我们的私有属性root是不需要的,因为"旋转"的性质导致我们的root始终在调整变化!
    */
    AVLNode<T> *root;
    bool taller = false;
public:
    AVLTREE()  root = nullptr; ;
    AVLNode<T> *&getroot()  return root; ;
    void insert(AVLNode<T>* &root, T data);  // 因为这里要递归地去进行插入操作,所以应该有一个当前root结点,非类的root
    bool deletenode(T data);
    bool search(AVLNode<T>* root, T data);
    void R_Rotate(AVLNode<T>* &_root);
    void L_Rotate(AVLNode<T>* &_root);
    // 定义了上面两种旋转方式,但实际上还有"双旋"的情况,但是双旋是可以分步完成的!分别是不同情况的失衡时的解决策略
    void LeftBalance(AVLNode<T>* &_root);
    void RightBalance(AVLNode<T>* &_root);
    // 展示是否有序!
    void MidOrderTraverse(AVLNode<T> *_root);
    int GetHeight(AVLNode<T> *_root);
;

template<typename T>
bool AVLTREE<T>::search(AVLNode<T>* _root, T data)
    // 这里和BSTREE的实现方式不一样,我们用递归的方式来寻找
    if(_root)
        if(_root->data == data)
            return true;
        
        else if(_root->data > data)
            return search(_root->left, data);
        
        else
            return search(_root->right, data);
        
    
    return false;


template<typename T>
void AVLTREE<T>::R_Rotate(AVLNode<T>* &_root)
    AVLNode<T> *left_sub_tree = _root->left;
    _root->left = left_sub_tree->right;
    left_sub_tree->right = _root;
    _root = left_sub_tree;


template<typename T>
void AVLTREE<T>::L_Rotate(AVLNode<T>* &_root)
    AVLNode<T>* right_sub_tree = _root->right;
    _root->right = right_sub_tree->left;
    right_sub_tree->left = _root;
    _root = right_sub_tree;


template<typename T>
void AVLTREE<T>::LeftBalance(AVLNode<T>* &_root)
    AVLNode<T> *leftsubtree = _root->left;
    AVLNode<T> *rightsubtree = leftsubtree->right;
    switch (leftsubtree->balance_factor)
    
    case LeftHigher:
        /*
        考虑这种情况,遇到第一个结点不平衡,且是因为左边过高才调用的LeftBalance函数。
        在此基础上自己的左子树根结点的平衡因子也是左高,就是我们的 朝左链成一条的不平衡情况
        此时只需做一个右旋就可,且旋转完两者均平衡!
        */
        _root->balance_factor = Equal;
        leftsubtree->balance_factor = Equal;
        R_Rotate(_root);
        break;
    
    case RightHigher:
        // 根据不同的情况先设置这些个结点的balance_factor
        switch (rightsubtree->balance_factor)
        
        case Equal:
            _root->balance_factor = Equal;
            leftsubtree->balance_factor = Equal;
            break;
        
        case LeftHigher:
            rightsubtree->balance_factor = leftsubtree->balance_factor = Equal;
            _root->balance_factor = RightHigher;
            break;
        
        case RightHigher:
            _root->balance_factor = rightsubtree->balance_factor = Equal;
            leftsubtree->balance_factor = LeftHigher;
            break;

        default:
            break;
        
        L_Rotate(_root->left);
        R_Rotate(_root);
        break;

    default:
        break;
    


template<typename T>
void AVLTREE<T>::RightBalance(AVLNode<T>* &_root)
    AVLNode<T> *rightsubtree = _root->right;
    AVLNode<T> *leftsubtree = rightsubtree->left;
    switch (rightsubtree->balance_factor)
    
    case RightHigher:
        _root->balance_factor = Equal;
        rightsubtree->balance_factor = Equal;
        L_Rotate(_root);
        break;
    
    case LeftHigher:
        // 根据不同的情况先设置这些个结点的balance_factor
        switch (leftsubtree->balance_factor)
        
        case Equal:
            _root->balance_factor = Equal;
            rightsubtree->balance_factor = Equal;
            break;
        
        case LeftHigher:
            _root->balance_factor = leftsubtree->balance_factor = Equal;
            rightsubtree->balance_factor = RightHigher;
            break;
        
        case RightHigher:
            rightsubtree->balance_factor = leftsubtree->balance_factor = Equal;
            _root->balance_factor = LeftHigher;
            break;

        default:
            break;
        
        R_Rotate(_root->right);
        L_Rotate(_root);
        break;

    default:
        break;
    


template<typename T>
void AVLTREE<T>::insert(AVLNode<T>* &_root, T data)
    if(!_root)                    // 用指针的引用可以直接连接到树上   因为指针的引用可以改变原父结点的孩子的地址指向!
        _root = new AVLNode<T>(data);
        taller = true;
    
    else
        if(_root->data > data)
            // _root->balance_factor += 1;
            insert(_root->left, data);
            if(taller)
                switch (_root->balance_factor)
                
                case RightHigher:
                    _root->balance_factor = Equal;
                    taller = false;
                    break;
                
                case Equal:
                    _root->balance_factor = LeftHigher;
                    taller = true;
                    break;

                case LeftHigher:
                    LeftBalance(_root);
                    taller = false;
                    break;

                default:
                    break;
                
            
        
        else
            // _root->balance_factor -= 1;
            insert(_root->right, data);
            if(taller)
                switch (_root->balance_factor)
                
                case LeftHigher:
                    _root->balance_factor = Equal;
                    taller = false;
                    break;

                case Equal:
                    _root->balance_factor = RightHigher;
                    taller = true;
                    break;
                
                case RightHigher:
                    RightBalance(_root);
                    taller = false;
                    break;

                default:
                    break;
                
            
        
    


template<typename T>
void AVLTREE<T>::MidOrderTraverse(AVLNode<T>* _root)
    if(_root)
        MidOrderTraverse(_root->left);
        cout << _root->data << " ";
        MidOrderTraverse(_root->right);
    


template<typename T>
int AVLTREE<T>::GetHeight(AVLNode<T>* _root)
    // 这里利用层序遍历【队列的方法】来进行求解高度
    if(!_root)
        return 0;
    queue<AVLNode<T> *> que;
    que.push(_root);
    int height = 0;
    while(!que.empty())
        for (size_t i = 0; i < que.size(); i ++)
            AVLNode<T> *temp = que.front();
            if(temp->left)
                que.push(temp->left);
            if(temp->right)
                que.push(temp->right);
            que.pop();
        
        height++;
    
    return height;


int main()
    AVLTREE<int> tree;
    AVLNode<int>* root = tree.getroot();
    vector<int> vec-4, 0, 7, 4, 9, -6, -1, -7;
    vector<int> vec20, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10;   // 用一个有序数组,对比BST和AVL树,性能差别就很明显了!
    // 在BST里会是一条长度为10[即树的高度]的链表,但在AVL中,树高仅为4!! 这显然好很多,搜索复杂度就牛逼起来了!!

    for(auto num : vec2)
        if(!tree.search(root, num))
            tree.insert(root, num);
        else
            cout << "It has been in the tree!" << endl;
        
    
    cout << "current root is:" << root->data << endl;
    cout << "height of the tree is:" << tree.GetHeight(root) << endl;
    tree.MidOrderTraverse(root);
    return 0;


以上是关于从BinarySearchTree到RedBlackTree之AVL的主要内容,如果未能解决你的问题,请参考以下文章

从BinarySearchTree到RedBlackTree之AVL

从BinarySearchTree到RedBlackTree之BST

从BinarySearchTree到RedBlackTree之BST

从BinarySearchTree到RedBlackTree之BST

javascript binarySearchTree由plastikaweb创建 - https://repl.it/@plastikaweb/binarySearchTree

BinarySearchTree-二叉搜索树