红黑树

红黑树(Red Black Tree) 是一种自平衡二叉查找树,典型用途是实现关联数组。

它是在1972年由Rudolf Bayer发明的,当时被称为平衡二叉B树(symmetric binary B-trees)。后来,在1978年被 Leo J. Guibas 和 Robert Sedgewick 修改为如今的“红黑树”。

约束性质:

    1、每一个节点非红即黑

    2、根节点永远为黑的

    3、每一个叶子节点都是黑色的(NLL,空节点)

    4、每个红色节点的两个子节点都是黑色的(从每个叶子到根的所有路径上不能有两个连续的红色节点)

    5、从任意一个节点到其每个叶子节点的所有路径都包含相同数目的黑色节点

    这些约束性质强制了红黑树的关键性质,从根到叶子节点的最长可能路径不多于最短可能路径的两倍长(第四五条),这条性质保证了这个树在高度上大致上时平衡的,因为操作比如插入、删除和查找某个值的最坏情况时间都要求与树的高度成比例。

    叶子节点不保存数据只作为树在此结束的标志。

    红黑色只需要每个节点中1 bit的存储空间

    红黑树在函数编程中也特别有用,在这里它们是最常用的持久数据结构之一,它们用来构造关联数组和集合,在突变之后它们能保持为以前的版本。除了O(log n)的时间之外,红黑树的持久版本对每次插入或删除需要O(log n)的空间。


当在对红黑树进行插入和删除等操作时,对树做了修改可能会破坏红黑树的性质。为了继续保持红黑树的性质,可以通过对结点进行重新着色,以及对树进行相关的旋转操作,即通过修改树中某些结点的颜色及指针结构,来达到对红黑树进行插入或删除结点等操作后继续保持它的性质或平衡的目的。

左旋
右旋

红黑树的修正:

    1、当前节点为红色,且父节点和叔父节点为红色,那么将父节点以及叔叔节点涂黑,将祖父节点涂红。

    2、当前节点为红色,并且是右子叶,父节点为红,且叔父节点为黑,那么以当前节点为基准进行左旋

    3、   当前节点为红色,并且为左子叶,父节点为红且叔父节点为黑,那么以父节点为基准进行右旋。交换祖父与父节点颜色


红黑树的插入

    插入节点为红色(如果我们插入的是黑色节点,那就违反了性质五,需要进行大规模调整)

    1、当插入节点是根节点时,直接涂黑插入

    2、当插入节点父节点为黑色时,直接插入

    其他情况先插入,再修正。


红黑树的删除

    1、当删除的元素为红时,对五条性质没有影响,直接删除

    2、当删除节点为根节点时,直接删除

    3、如果被删元素只有一个子节点,则其子节点顶替其位置,只交换值,不换颜色

    4、如果有两个子节点,这时候就可以把被删除元素的右支的最小节点(被删除元素右支的最左边的节点)和被删除元素互换,我们把被删除元素右支的最左边的节点称之为后继节点(后继元素)、只交换值,不换颜色

    然后对其进行修正


##Code

```

package com.gdufe.binarysearchtree;

import java.io.File;

import java.util.Scanner;

public class RedBlackTree<Key extends Comparable<Key>, Value> {

    Node root; // 维护根节点

    final static boolean RED = true;

    final static boolean BLACK = false;

    class Node { // 二叉树的结点

        Key key;

        Value value;

        boolean color;

        Node left, right;

        public Node(Key key, Value value, boolean color) { // 初始化结点

            this.key = key;

            this.value = value;

            this.color = color;

        }

    }

    public Value get(Key key) {

        return get(root, key);

    }

    // 右旋

    public Node rotateRight(Node h) {

        Node x = h.left;

        h.left = x.right;

        x.right = h;

        x.color = h.color;

        h.color = RED;

        return x;

    }

    // 左旋

    public Node rotateLeft(Node h) {

        Node x = h.right;

        h.right = x.left;

        x.left = h;

        x.color = h.color;

        h.color = RED;

        return x;

    }

    // 变色处理

    public void flipColors(Node h) {

        h.left.color = BLACK;

        h.right.color = BLACK;

        h.color = RED;

    }

    public boolean isRed(Node x){

        if(x==null) return false;

        else return x.color;

    }

    public Value get(Node x, Key key) {

        if (x == null)

            return null;

        int cmp = key.compareTo(x.key);

        if (cmp < 0)

            return get(x.left, key);

        else if (cmp > 0)

            return get(x.right, key);

        else

            return x.value;

    }

    public void put(Key key, Value value) {

        root = put(root, key, value);

        root.color = BLACK;

    }

    public Node put(Node x, Key key, Value value) {

        if (x == null)

            return new Node(key, value, RED); // 添加的结点链接为红色

        int cmp = key.compareTo(x.key);

        if (cmp < 0)

            x.left = put(x.left, key, value);

        else if (cmp > 0)

            x.right = put(x.right, key, value);

        else {

            x.value = value;

        }

        // 判断是否需要左旋,右旋,变色操作

        if (x != null) {

            if (!isRed(x.left) && isRed(x.right))

                x = rotateLeft(x);

            if (isRed(x.left) && isRed(x.left.left))

                x = rotateRight(x);

            if (isRed(x.left ) && isRed(x.right))

                flipColors(x);

        }

        return x;

    }

    public static void main(String[] args) throws Exception {

        Scanner input = new Scanner(new File("data_BST.txt"));

        RedBlackTree<String, Integer> bst = new RedBlackTree<String, Integer>();

        while (input.hasNext()) {

            String key = input.next();

            int value = input.nextInt();

            bst.put(key, value);

        }

        System.out.println(bst.get("H"));

        System.out.println(bst.get("B"));

    }

}

```

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容