2018-10-08 红黑树

2018-10-08  本文已影响0人  范正辰

红黑树实现

1、五个性质

(1)根节点黑色

(2) 只有红色和黑色节点

(3) 红色节点相邻节点黑色

(4) 每个节点到任意子树叶子节点黑色节点个数相同

(5) 叶子节点黑色

2、插入

插入节点颜色为红色

(1) 根节点为null 直接赋值

(2) 插入节点父亲节点为黑色结束

(3) 插入节点父亲节点颜色为红色

1、叔叔节点颜色为红色

                 b               r           
                / \             / \
               r   r    ->     b   b
              /               / 
            (r)             (r)      

2、叔叔节点为黑色

                    b             b
                   / \           / \
                  r   b  ->   (r)   r
                 /                   \
               (r)                    b
or
                       b             b
                      / \           / \
                     r   b  ->    (r)  b
                      \           /
                      (r)        r
                

我的实现

if (rt->color == RED && par->color == RED) {
        Node *grand = par->parent;
        bool fc = (grand->ch[1] == par);
        Node *uncle = grand->ch[fc ^ 1];

        /*
         *        b               r
         *       / \             / \
         *      r   r    ->     b   b
          *    /               /
         *   (r)             (r)
         *
         */

        if (uncle->color == RED) {
            uncle->color = par->color = BLACK;
            grand->color = RED;
        } else {
            bool c = (par->ch[1] == rt);

            /**
             *       b             b
             *      / \           / \
             *     r   b  ->   (r)   r
             *    /                   \
             *  (r)                    b
             */
            if (c == fc) {
                par->color = BLACK;
                grand->color = RED;
                rotate(par);
            } else {
                /**
                 *      b             b
                 *     / \           / \
                 *    r   b  ->    (r)  b
                 *     \           /
                 *     (r)        r
                 */
                rotate(rt);
                pushUp(rt->ch[c ^ 1]);
            }
        }
    } 

3、删除

找到前继节点或者后继节点,颜色不变,值互换下,然后开始删前继或者后继节点,这个时候删除其实有9种情况

(1) 根节点直接删

(2) 红色节点直接删

(3) 删除节点父亲节点为红色,父亲变黑

(4) 子节为红色的,变黑

然后就是红黑树里面最蛋疼的地方了一共五种情况,我们已经删除了节点,要开始平衡红黑树,看代码

/**
 * delete balance
 * @param rt
 */
void RedBlackTree::balance(Node *rt) {
    if (rt == root) {
        return;
    }


    Node *par = rt->parent;
    bool c = par->isRightChild();
    Node *brother = par->ch[c ^ 1];



    /**
     *
     *    B                 (B)
     *   / \                / \
     * (B)  B     ----->   B   R
     *     / \                / \
     *    B   B              B   B
     */
    if (par->color == BLACK && brother->color == BLACK && brother->ch[c] == BLACK
        && brother->ch[c ^ 1] == BLACK) {
        brother->color = RED;
        balance(par);
        return;
    }



    /**
     *        B1            B3
     *       / \   ---->   /
     *    (B)2  R3        R1
     *                    /
     *                  (B)2
     *
     *   grantee that brother is not red in the following
     */
    if (brother->color == RED) {
        std::swap(brother->color, par->color);
        rotate(brother);
    }



    /**
     *     R                 (B)
     *    / \                / \
     *  (B)  B    ----->    B   R
     *      / \                / \
     *     B   B              B   B
     */
    assert(brother->color == BLACK);
    brother = par->ch[c ^ 1];
    if (par->color == RED && brother->color == BLACK && brother->ch[c] == BLACK
        && brother->ch[c ^ 1] == BLACK) {
        std::swap(par->color, brother->color);
        balance(par);
        return;
    }

    /**
     *      X              X
     *     / \            / \
     *   (B)  B2   -->  (B)  B1
     *       / \              \
     *      R1  B3             R2
     *                          \
     *                           B3
     */

    Node *left = brother->ch[c];
    if (left->color == RED) {
        std::swap(brother->color, left->color);
        rotate(left);
    }


    /**
     *       X1                     X2
     *      / \                    /  \
     *    (B)  B2     ----->      B1   B4
     *        / \                / \
     *       X3  R4            (B)  X3
     */
    Node *right = brother->ch[c ^ 1];
    if (right->color == RED) {
        std::swap(brother->color, par->color);
        right->color = BLACK;
        rotate(brother);
    }

}

我的旋转代码 当前节点往父亲节点旋,这个可以根据自己的喜好来

void RedBlackTree::rotate(Node *rt) {

    Node *father = rt->parent;
    Node *grand = father->parent;


    bool c = (father->ch[1] == rt);
    bool fc = (grand->ch[1] == father);

    Node *&ch = rt->ch[c ^ 1];
    father->ch[c] = ch;
    if (ch != nullNode) {
        ch->parent = father;
    }
    ch = father;
    ch->parent = rt;


    rt->parent = grand;
    grand->ch[fc] = rt;

    if (father == root) {
        root = rt;
    }

    maintain(father);
    maintain(rt);

}
上一篇 下一篇

猜你喜欢

热点阅读