TreeMap是一个有序的key-value集合,基于红黑树(Red-Black tree)实现。该映射根据其键的自然顺序进行排序,或者根据创建时提供的Comparator进行排序、

对于TreeMap而言,每个Entry都被当成“红黑树”的一个节点对待,示例如下:

public class TreeMapTest {
    public static void main(String[] args) {
        TreeMap<String, Float> map = new TreeMap<String, Float>();
        map.put("B", 88.0f);
        map.put("C", 95.0f);
        map.put("A", 90.0f);

        System.out.println(map);
    }
}

当程序执行map.put(“B”, 88.0f);时,系统将直接把 “B”-88.0f 这个Entry放入Map中,这个Entry就是该“红黑树”的根节点。接着程序执行 map.put(“C”, 95.0f); 时,会将 “C”-95.0f 作为新节点添加到已有的红黑树中。

以后每向TreeMap中放入一个key-value对,系统都需要将该Entry当成一个新节点,添加到已有红黑树中,通过这种方式就可以保证TreeMap中所有key总是由小到大地排列。例如,输出上面程序,将看到如下结果(所有key由小到大地排列)。

{A=90.0, B=88.0, C=95.0}

对于TreeMap而言,由于它底层采用一颗“红黑树”来保存集合中的Entry,这意味着TreeMap添加元素、取出元素的性能都比HashMap低。当TreeMap添加元素时,需要通过循环找到新增Entry的插入位置,因此比较耗性能;当从TreeMap中取出元素时,需要通过循环才能找到合适的Entry,也比较耗性能。但TreeMap、TreeSet相比HashMap、HashSet的优势在于:TreeMap中的所有Entry总是按key根据指定排序规则保存有序状态,TreeSet中的所有元素总是根据指定排序规则保存有序状态。

红黑树是一种自平衡二叉查找树,树中每个节点的值,都大于或等于在它的左子树中的所有节点的值,并且小于或等于在它的右子树中的所有节点的值,这确保红黑树运行时可以快速地在树中查找和定位的所需节点。

对于TreeMap集合而言,其关键就是put(K key, V value), 该方法实现了将Entry放入TreeMap的Entry链,并保证该Entry链总是处于有序状态。下面是该方法的源代码。

public V put(K key, V value) {
    // 先以 t 保存链表的 root 节点
    Entry<K,V> t = root;
    // 如果 t==null,表明是一个空链表,即该 TreeMap 里没有任何 Entry
    if (t == null) {
        // 将新的key-value创建一个 Entry,并将该 Entry 作为 root
        root = new Entry<K,V>(key, value, null);
    //设置该Map集合的size为1,代表包含一个Entry
        size = 1;
    //记录修改次数为1
        modCount++;
        return null;
    }
    int cmp;
    Entry<K,V> parent;
    // split comparator and comparable paths
    Comparator<? super K> cpr = comparator;
    // 如果比较器 cpr 不为 null,即表明采用定制排序
    if (cpr != null) {
        do {
            // 使用 parent 上次循环后的 t 所引用的 Entry
            parent = t;
        // 拿新插入的key和t的key进行比较
            cmp = cpr.compare(key, t.key);
        // 如果新插入的key小于t的key,t等于t的左边节点
            if (cmp < 0)
                t = t.left;
        // 如果新插入的key大于t的key,t等于t的右边节点
            else if (cmp > 0)
                t = t.right;
        // 如果两个key相等,新value覆盖原有的value,并返回原有的value
            else
                return t.setValue(value);
        } while (t != null);
    }
    else {
        if (key == null)
            throw new NullPointerException();
        Comparable<? super K> k = (Comparable<? super K>) key;
        do {
        // 使用parent上次循环后的t所引用的Entry
            parent = t;
        // 拿新插入的key和t的key进行比较
            cmp = k.compareTo(t.key);
        // 如果新插入的key小于t的key,t等于t的左边节点
            if (cmp < 0)
                t = t.left;
        // 如果新插入的key大于t的key,t等于t的右边节点
            else if (cmp > 0)
                t = t.right;
        // 如果两个key相等,新value覆盖原有的value,并返回原有的value
            else
                return t.setValue(value);
        } while (t != null);
    }
    // 将新插入的节点作为parent节点的子节点
    Entry<K,V> e = new Entry<K,V>(key, value, parent);
    // 如果新插入key小于parent的key,则e作为parent的左子节点
    if (cmp < 0)
        parent.left = e;
    // 如果新插入key小于parent的key,则e作为parent的右子节点
    else
        parent.right = e;
    // 修复红黑树
    fixAfterInsertion(e);          //①
    size++;
    modCount++;
    return null;
}

上面程序中的粗体字代码就是实现排序二叉树的关键算法。每当程序希望添加新节点时,总是从树的跟节点开始比较,即将跟节点当成当前节点。如果新增节点大于当前节点且当前节点的右子节点存在,则以右子节点作为当前节点;如果新增节点小于当前节点且当前节点的左子节点存在,则以左子节点作为当前节点;如果新增节点等于当前节点,则用新增节点覆盖当前节点,并结束循环–直到找到某个节点的左、右子节点不存在,将新节点添加为该节点的子节点。如果新节点比该节点大,则添加其为右子节点;如果新节点比该节点小,则添加其为左子节点。

当TreeMap根据key来取出value时,TreeMap对应的方法如下。

public V get(Object key) {
    // 根据指定key取出对应的Entry
    Entry<K,V> p = getEntry(key);
    // 返回该Entry所包含的value
    return (p==null ? null : p.value);
}

从上面程序代码可以看出,get(Object key)方法实质上是由getEntry()方法实现的。
这个getEntry()方法的代码如下。

final Entry<K,V> getEntry(Object key) {
    // 如果comparator不为null,表明程序采用定制排序
    if (comparator != null)
    // 调用getEntryUsingComparator方法取出对应的key
        return getEntryUsingComparator(key);
    // 如果key形参的值为null,抛出NullPointerException异常
    if (key == null)
        throw new NullPointerException();
// 将key强制类型转换为Comparable实例
Comparable<? super K> k = (Comparable<? super K>) key;
    // 从树的根节点开始
    Entry<K,V> p = root;
    while (p != null) {
        // 拿key与当前节点的key进行比较
        int cmp = k.compareTo(p.key);
    // 如果key小于当前节点的key,向左子树搜索
        if (cmp < 0)
            p = p.left;
    // 如果key大于当前节点的key,向右子树搜索
        else if (cmp > 0)
            p = p.right;
    // 如果既不大于也不小于,就是找到了目标Entry
        else
            return p;
    }
    return null;
}

上面的getEntry(Object obj)方法也是充分利用排序二叉树的特征来搜索目标Entry。程序依然从二叉树的根节点开始,如果被搜索节点大于当前节点,程序向“右子树”搜索;如果被搜索节点小于当前节点,程序向“左子树”搜索;如果相等,那就是找到了指定节点。

当TreeMap里的comparator!=null,即表明该TreeMap采用了定制排序。在采用定制排序的方式下,TreeMap采用getEntryUsingComparator(key)方法来根据key获取Entry。下面是该方法的代码。

final Entry<K,V> getEntryUsingComparator(Object key) {
    K k = (K) key;
    // 获取该TreeMap的comparator
        Comparator<? super K> cpr = comparator;
        if (cpr != null) {
        //从根节点开始
            Entry<K,V> p = root;
            while (p != null) {
            // 拿key与当前节点的key进行比较
                int cmp = cpr.compare(k, p.key);
        // 如果key小于当前节点的key,向“左子树”搜索
                if (cmp < 0)
                    p = p.left;
        // 如果key大于当前节点的key,向“右子树”搜索
                else if (cmp > 0)
                    p = p.right;
        // 如果既不大于也不小于,就是找到了目标Entry
                else
                    return p;
            }
        }
        return null;
}

其实getEntry,getEntryUsingComparator这2个方法的实现思路完全类似,只是前者对自然排序的TreeMap获取有效,后者对定制排序的TreeMap有效。

通过上面源代码的分析不难看出,TreeMap这个工具类的实现其实很简单。或者说,从内部结构来看,TreeMap本质上就是一颗“红黑树”,而TreeMap的每个Entry就是该红黑树的一个节点。

TreeMap实现原理及源码分析的更多相关文章

  1. TreeMap实现原理及源码分析之JDK8

    转载 Java 集合系列12之 TreeMap详细介绍(源码解析)和使用示例 一.TreeMap 简单介绍 什么是Map? 在数组中我们通过数组下标来对数组内容进行索引的,而在Map中我们通过对象来对 ...

  2. HashMap实现原理及源码分析之JDK8

    继续上回HashMap的学习 HashMap实现原理及源码分析之JDK7 转载 Java8源码-HashMap  基于JDK8的HashMap源码解析  [jdk1.8]HashMap源码分析 一.H ...

  3. OpenCV学习笔记(27)KAZE 算法原理与源码分析(一)非线性扩散滤波

    http://blog.csdn.net/chenyusiyuan/article/details/8710462 OpenCV学习笔记(27)KAZE 算法原理与源码分析(一)非线性扩散滤波 201 ...

  4. ConcurrentHashMap实现原理及源码分析

    ConcurrentHashMap实现原理 ConcurrentHashMap源码分析 总结 ConcurrentHashMap是Java并发包中提供的一个线程安全且高效的HashMap实现(若对Ha ...

  5. HashMap和ConcurrentHashMap实现原理及源码分析

    HashMap实现原理及源码分析 哈希表(hash table)也叫散列表,是一种非常重要的数据结构,应用场景及其丰富,许多缓存技术(比如memcached)的核心其实就是在内存中维护一张大的哈希表, ...

  6. (转)ReentrantLock实现原理及源码分析

    背景:ReetrantLock底层是基于AQS实现的(CAS+CHL),有公平和非公平两种区别. 这种底层机制,很有必要通过跟踪源码来进行分析. 参考 ReentrantLock实现原理及源码分析 源 ...

  7. 【转】HashMap实现原理及源码分析

    哈希表(hash table)也叫散列表,是一种非常重要的数据结构,应用场景极其丰富,许多缓存技术(比如memcached)的核心其实就是在内存中维护一张大的哈希表,而HashMap的实现原理也常常出 ...

  8. 【OpenCV】SIFT原理与源码分析:DoG尺度空间构造

    原文地址:http://blog.csdn.net/xiaowei_cqu/article/details/8067881 尺度空间理论   自然界中的物体随着观测尺度不同有不同的表现形态.例如我们形 ...

  9. 《深入探索Netty原理及源码分析》文集小结

    <深入探索Netty原理及源码分析>文集小结 https://www.jianshu.com/p/239a196152de

随机推荐

  1. 20145303刘俊谦 《Java程序设计》第三周学习总结

    20145303刘俊谦 <Java程序设计>第三周学习总结 教材学习内容总结 1.类与对象: 类:对现实生活中事物的描述,定义类时用关键词class 对象:这类事物实实在在存在的个体,利用 ...

  2. [CF1042F]Leaf Sets

    题意:给定一棵$n$个点的树,将叶子节点分为数个集合使集合里点对最长距离不超过$k$,求最少集合数.($n\le1000000$) 首先我们可以想到,这道题并不是让你构造最优方案,因为只要把所有叶子节 ...

  3. js 自定义事件观察者模式(发布/订阅)

    /* * 示例: * Event.create("namespace1").listen('click', function(a){ * console.log(a); * }); ...

  4. JAVA多线程本质分析

    多线程是Java开发中的重中之重,其重要性和难度,可见一斑.掌握并精通多线程开发,是每一个程序员的必修之课.哪怕中间的过程很痛苦,只要坚持了,并最终豁然开朗了,都是一种升华. 多线程的优化:合理利用C ...

  5. git gc内存错误的解决方案

    Auto packing the repository for optimum performance. You may alsorun "git gc" manually. Se ...

  6. Elasticsearch之几个重要的分词器

    前提 什么是倒排索引? Elasticsearch之分词器的作用 Elasticsearch之分词器的工作流程 Elasticsearch之停用词 Elasticsearch之中文分词器 Elasti ...

  7. 生信实验室收集---Dana Pe'er Lab

    Dana Pe'er Lab of Computational Systems Biology Dana Pe'er是哥伦比亚大学生物科学系的副教授,被认为是计算系统生物学的主要研究人员之一.Dana ...

  8. html-w3c规范及常见标签

    W3C提倡的web结构: 内容(HTML)与表现(css样式)分离 内容(HTML)与行为(JS)分离 HTML内容结构要求语义化 基本规范: 标签名和属性名称必须小写 HTML标签必须关闭 属性值必 ...

  9. 贯众云平台脚本编写之判断、循环以及shell命令的使用

    最近使用贯众云平台工具写脚本,进行Ui界面的自动化测试.刚开始接触,确实碰到不少的问题,稍微难点的就是判断语句,循环语句以及shell命令的使用,尤其是对于咱们测试这样比较少接触代码的人来说. 其实吧 ...

  10. Android安装过程出现问题

    Android安装过程出现问题 一.Eclipse 中 Emulator Control 不能用问题 在官方文档中发现问题所在(官方文档说明),在最后一行“The Emulator Control t ...