• 大名鼎鼎的红黑树,你get了么?2-3树 绝对平衡 右旋转 左旋转 颜色反转


     前言

      11.1新的一月加油!这个购物狂欢的季节,一看,已囊中羞涩!赶紧来恶补一下红黑树和2-3树吧!红黑树真的算是大名鼎鼎了吧?即使你不了解它,但一定听过吧?下面跟随我来揭开神秘的面纱吧!

      一、2-3树

      1、抢了红黑树的光环?

      今天的主角是红黑树,是无疑的,主角光环在呢!那2-3树又是什么鬼呢?学习2-3树不仅对理解红黑树有帮助,对理解B类树,也是有巨大帮助的,所以学习2-3树很必要!

      2、基本性质

      2-3树满足二分搜索树的基本性质,但节点可以存放一个元素或两个元素!如下图,就是2-3树:

      

      说明:2-3树一颗绝对平衡的树(绝对平衡:对于任意一个节点,左右子树高度相同)

      3、维护绝对平衡

      2-3树在插入过程中如何维护绝对平衡呢?进行画图演示,实在有点不好画,如下图:

      

      说明:

      1、不能将新节点插入到空节点

        因为那样如上图,就不满足绝对平衡了,所以可以将37和42合并,2-3支持3节点。

      2、不支持4节点,进行拆分

        再插入12时,也不能插入空节点,也要合并,但2-3树不支持4节点,所以进行进行拆分。

      3、子节点达到3节点,合并到父节点

        再依次插入18、6,达到4节点,进行拆分,但不符合绝对平衡了怎么办?将12和37合并,就形成了最后3节点的图了

      总结:讲到这里,应该对2-3树如何维护绝对平衡,应该了解了吧?理解2-3树,对于再理解红黑树,是非常有帮助的,其实,它们有等价性的,接下来会说明的。

      二、红黑树

      1、红黑树和2-3树的等价性

      也想达到像2-3树那样的绝对平衡,但2-3树的实现比较麻烦,所以产生了红黑树;那么,红黑树和2-3树有怎么样的等价性呢?如下图:

      

      说明:红黑树最开始想用红线区别b、c,但实现起来比较困难,然后用红黑来表示节点,就比较好实现了!

      红黑树和2-3树总体对比图,可以参考一下:

      

      2、红黑树5个重要性质

      1、引自《算法导论》

      红黑树有五个重要性质,引自算法界一本圣洁《算法导论》中的内容,如下:

      

      是不是看着有点晕,下面我进行解释。

      2、5个重要性质

      1、每一个节点或者红色的,或是黑色的

      2、根节点是黑色的

      3、每一个叶子节点(最后的空节点)是黑色的

      4、如果一个节点是红色的,那么它的孩子节点都是黑色的

      5、从任意节点到叶子节点,经过的黑色节点是一样的

      

      解释:最重要的性质是第五条,前4条在理解2-3树之后,就很好理解了,第5条性质说明了:红黑树是保持“黑平衡”的二叉树;

    严格意义上来说,红黑树不是平衡二叉树,最大高度:2logn,但是时间复杂度仍然是O(logn),因为2是常数,但比AVL树查询要稍微慢一些。

      三、红黑树添加元素

      红黑树添加元素,比较繁琐,因为要保持上面的五个性质,要不然就不是红黑树了;

      1、保持根节点为节点

      红黑树的节点类也可以从二分搜索树上进行修改,但要新增“color”成员变量,来标注节点颜色,节点类如下:

    复制代码
    template<typename Key, typename Value>
    class RBTree {
    private:
        static const bool RED = true;
        static const bool BLACK = false;
    
        struct Node {
            Key key;
            Value value;
            Node *left;
            Node *right;
            bool color;
    
            Node(Key key, Value value) {
                this->key = key;
                this->value = value;
                this->left = this->right = nullptr;
                color = RED;  //默认初始化为红色
            }
    
            Node(Node *node) {
                this->key = node->key;
                this->value = node->value;
                this->left = node->left;
                this->right = node->right;
                this->color = node->color;
            }
        };
    
        Node *root;
        int size;
    }
    复制代码

      因为红黑树性质1要求根节点为黑色,所以要保持根节点为黑色;

      2、左旋转

      像AVL树一样,红黑树也需要左旋和右旋,如下图就需要左旋转,因为“红色节点是左倾斜的”:

      

      说明:图中黑色字体标识黑色节点,红色表示红色节点,并演示了旋转过程,最后还要改变节点颜色。

      3、左旋转代码实现

      代码如下:

      

    复制代码
    Node *leftRotate(Node *node) {
            Node *x = node->right;
            node->right = x->left;
            x->left = node;
    
            x->color = node->color;
            node->color = RED;
    
            return x;
        }
    复制代码

      4、颜色反转

      下面这种情况就需要颜色反转,如下图:

      

      

      5、颜色反转代码实现

      代码如下:

    void flipColors(Node *node) {
            node->color = RED;
            node->left->color = BLACK;
            node->right->color = BLACK;
        }

      6、右旋转

      下面情况需要右旋转,如下图:

      

        旋转之后,如下图:

      

       7、右旋转代码如下

      代码如下:

      

    复制代码
    Node *rightRotate(Node *node) {
            Node *x = node->left;
            node->left = x->right;
            x->right = node;
    
            x->color = node->color;
            node->color = RED;
    
            return x;
        }
    复制代码

      8、总体流程图

      

      9、总体代码

      总体代码如下,供参考和学习:

     View Code

      总结  

      面试时99.9%不会让手写一下红黑树的添加过程,除非你面试算法工程师,那就打扰了!主要理解红黑树的性质、左旋和右旋等。

      欢迎点赞和评论,感谢支持!

  • 相关阅读:
    如何写好电子邮件
    【SQL查询】正则表达式匹配字符串_regexp_like/substr/instr/replace
    python_安装第三方库
    【SQL查询】分区查询Over
    【SQL查询】树结构查询
    【PL/SQL编程】块结构
    STL初探
    普及C组第四题(8.18)
    来纪中的的第18天
    来纪中的第十七天
  • 原文地址:https://www.cnblogs.com/ExMan/p/9894890.html
Copyright © 2020-2023  润新知