欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

死磕 java集合之TreeMap源码分析(一)

程序员文章站 2022-06-22 13:08:56
死磕 java集合之TreeMap源码分析(一) 红黑树是什么?有什么特性? 它的时间复杂度是多少? 它跟SortedMap有什么区别和联系? 它的左旋、右旋是怎么玩的? ......

欢迎关注我的公众号“彤哥读源码”,查看更多源码系列文章, 与彤哥一起畅游源码的海洋。

简介

treemap使用红黑树存储元素,可以保证元素按key值的大小进行遍历。

继承体系

死磕 java集合之TreeMap源码分析(一)

treemap实现了map、sortedmap、navigablemap、cloneable、serializable等接口。

sortedmap规定了元素可以按key的大小来遍历,它定义了一些返回部分map的方法。

public interface sortedmap<k,v> extends map<k,v> {
    // key的比较器
    comparator<? super k> comparator();
    // 返回fromkey(包含)到tokey(不包含)之间的元素组成的子map
    sortedmap<k,v> submap(k fromkey, k tokey);
    // 返回小于tokey(不包含)的子map
    sortedmap<k,v> headmap(k tokey);
    // 返回大于等于fromkey(包含)的子map
    sortedmap<k,v> tailmap(k fromkey);
    // 返回最小的key
    k firstkey();
    // 返回最大的key
    k lastkey();
    // 返回key集合
    set<k> keyset();
    // 返回value集合
    collection<v> values();
    // 返回节点集合
    set<map.entry<k, v>> entryset();
}

navigablemap是对sortedmap的增强,定义了一些返回离目标key最近的元素的方法。

public interface navigablemap<k,v> extends sortedmap<k,v> {
    // 小于给定key的最大节点
    map.entry<k,v> lowerentry(k key);
    // 小于给定key的最大key
    k lowerkey(k key);
    // 小于等于给定key的最大节点
    map.entry<k,v> floorentry(k key);
    // 小于等于给定key的最大key
    k floorkey(k key);
    // 大于等于给定key的最小节点
    map.entry<k,v> ceilingentry(k key);
    // 大于等于给定key的最小key
    k ceilingkey(k key);
    // 大于给定key的最小节点
    map.entry<k,v> higherentry(k key);
    // 大于给定key的最小key
    k higherkey(k key);
    // 最小的节点
    map.entry<k,v> firstentry();
    // 最大的节点
    map.entry<k,v> lastentry();
    // 弹出最小的节点
    map.entry<k,v> pollfirstentry();
    // 弹出最大的节点
    map.entry<k,v> polllastentry();
    // 返回倒序的map
    navigablemap<k,v> descendingmap();
    // 返回有序的key集合
    navigableset<k> navigablekeyset();
    // 返回倒序的key集合
    navigableset<k> descendingkeyset();
    // 返回从fromkey到tokey的子map,是否包含起止元素可以自己决定
    navigablemap<k,v> submap(k fromkey, boolean frominclusive,
                             k tokey,   boolean toinclusive);
    // 返回小于tokey的子map,是否包含tokey自己决定
    navigablemap<k,v> headmap(k tokey, boolean inclusive);
    // 返回大于fromkey的子map,是否包含fromkey自己决定
    navigablemap<k,v> tailmap(k fromkey, boolean inclusive);
    // 等价于submap(fromkey, true, tokey, false)
    sortedmap<k,v> submap(k fromkey, k tokey);
    // 等价于headmap(tokey, false)
    sortedmap<k,v> headmap(k tokey);
    // 等价于tailmap(fromkey, true)
    sortedmap<k,v> tailmap(k fromkey);
}

存储结构

死磕 java集合之TreeMap源码分析(一)

treemap只使用到了红黑树,所以它的时间复杂度为o(log n),我们再来回顾一下红黑树的特性。

(1)每个节点或者是黑色,或者是红色。

(2)根节点是黑色。

(3)每个叶子节点(nil)是黑色。(注意:这里叶子节点,是指为空(nil或null)的叶子节点!)

(4)如果一个节点是红色的,则它的子节点必须是黑色的。

(5)从一个节点到该节点的子孙节点的所有路径上包含相同数目的黑节点。

源码解析

属性

/**
 * 比较器,如果没传则key要实现comparable接口
 */
private final comparator<? super k> comparator;

/**
 * 根节点
 */
private transient entry<k,v> root;

/**
 * 元素个数
 */
private transient int size = 0;

/**
 * 修改次数
 */
private transient int modcount = 0;

(1)comparator

按key的大小排序有两种方式,一种是key实现comparable接口,一种方式通过构造方法传入比较器。

(2)root

根节点,treemap没有桶的概念,所有的元素都存储在一颗树中。

entry内部类

存储节点,典型的红黑树结构。

static final class entry<k,v> implements map.entry<k,v> {
    k key;
    v value;
    entry<k,v> left;
    entry<k,v> right;
    entry<k,v> parent;
    boolean color = black;
}

构造方法

/**
 * 默认构造方法,key必须实现comparable接口 
 */
public treemap() {
    comparator = null;
}

/**
 * 使用传入的comparator比较两个key的大小
 */
public treemap(comparator<? super k> comparator) {
    this.comparator = comparator;
}
    
/**
 * key必须实现comparable接口,把传入map中的所有元素保存到新的treemap中 
 */
public treemap(map<? extends k, ? extends v> m) {
    comparator = null;
    putall(m);
}

/**
 * 使用传入map的比较器,并把传入map中的所有元素保存到新的treemap中 
 */
public treemap(sortedmap<k, ? extends v> m) {
    comparator = m.comparator();
    try {
        buildfromsorted(m.size(), m.entryset().iterator(), null, null);
    } catch (java.io.ioexception cannothappen) {
    } catch (classnotfoundexception cannothappen) {
    }
}

构造方法主要分成两类,一类是使用comparator比较器,一类是key必须实现comparable接口。

其实,笔者认为这两种比较方式可以合并成一种,当没有传comparator的时候,可以用以下方式来给comparator赋值,这样后续所有的比较操作都可以使用一样的逻辑处理了,而不用每次都检查comparator为空的时候又用comparable来实现一遍逻辑。

// 如果comparator为空,则key必须实现comparable接口,所以这里肯定可以强转
// 这样在构造方法中统一替换掉,后续的逻辑就都一致了
comparator = (k1, k2) -> ((comparable<? super k>)k1).compareto(k2);

get(object key)方法

获取元素,典型的二叉查找树的查找方法。

public v get(object key) {
    // 根据key查找元素
    entry<k,v> p = getentry(key);
    // 找到了返回value值,没找到返回null
    return (p==null ? null : p.value);
}

final entry<k,v> getentry(object key) {
    // 如果comparator不为空,使用comparator的版本获取元素
    if (comparator != null)
        return getentryusingcomparator(key);
    // 如果key为空返回空指针异常
    if (key == null)
        throw new nullpointerexception();
    // 将key强转为comparable
    @suppresswarnings("unchecked")
    comparable<? super k> k = (comparable<? super k>) key;
    // 从根元素开始遍历
    entry<k,v> p = root;
    while (p != null) {
        int cmp = k.compareto(p.key);
        if (cmp < 0)
            // 如果小于0从左子树查找
            p = p.left;
        else if (cmp > 0)
            // 如果大于0从右子树查找
            p = p.right;
        else
            // 如果相等说明找到了直接返回
            return p;
    }
    // 没找到返回null
    return null;
}
    
final entry<k,v> getentryusingcomparator(object key) {
    @suppresswarnings("unchecked")
    k k = (k) key;
    comparator<? super k> cpr = comparator;
    if (cpr != null) {
        // 从根元素开始遍历
        entry<k,v> p = root;
        while (p != null) {
            int cmp = cpr.compare(k, p.key);
            if (cmp < 0)
                // 如果小于0从左子树查找
                p = p.left;
            else if (cmp > 0)
                // 如果大于0从右子树查找
                p = p.right;
            else
                // 如果相等说明找到了直接返回
                return p;
        }
    }
    // 没找到返回null
    return null;
}

(1)从root遍历整个树;

(2)如果待查找的key比当前遍历的key小,则在其左子树中查找;

(3)如果待查找的key比当前遍历的key大,则在其右子树中查找;

(4)如果待查找的key与当前遍历的key相等,则找到了该元素,直接返回;

(5)从这里可以看出是否有comparator分化成了两个方法,但是内部逻辑一模一样,因此可见笔者comparator = (k1, k2) -> ((comparable<? super k>)k1).compareto(k2);这种改造的必要性。


我是一条美丽的分割线,前方高能,请做好准备。


特性再回顾

(1)每个节点或者是黑色,或者是红色。

(2)根节点是黑色。

(3)每个叶子节点(nil)是黑色。(注意:这里叶子节点,是指为空(nil或null)的叶子节点!)

(4)如果一个节点是红色的,则它的子节点必须是黑色的。

(5)从一个节点到该节点的子孙节点的所有路径上包含相同数目的黑节点。

左旋

左旋,就是以某个节点为支点向左旋转。

死磕 java集合之TreeMap源码分析(一)

整个左旋过程如下:

(1)将 y的左节点 设为 x的右节点,即将 β 设为 x的右节点;

(2)将 x 设为 y的左节点的父节点,即将 β的父节点 设为 x;

(3)将 x的父节点 设为 y的父节点;

(4)如果 x的父节点 为空节点,则将y设置为根节点;如果x是它父节点的左(右)节点,则将y设置为x父节点的左(右)节点;

(5)将 x 设为 y的左节点;

(6)将 x的父节点 设为 y;

让我们来看看treemap中的实现:

/**
 * 以p为支点进行左旋
 * 假设p为图中的x
 */
private void rotateleft(entry<k,v> p) {
    if (p != null) {
        // p的右节点,即y
        entry<k,v> r = p.right;
        
        // (1)将 y的左节点 设为 x的右节点
        p.right = r.left;
        
        // (2)将 x 设为 y的左节点的父节点(如果y的左节点存在的话)
        if (r.left != null)
            r.left.parent = p;

        // (3)将 x的父节点 设为 y的父节点
        r.parent = p.parent;

        // (4)...
        if (p.parent == null)
            // 如果 x的父节点 为空,则将y设置为根节点
            root = r;
        else if (p.parent.left == p)
            // 如果x是它父节点的左节点,则将y设置为x父节点的左节点
            p.parent.left = r;
        else
            // 如果x是它父节点的右节点,则将y设置为x父节点的右节点
            p.parent.right = r;

        // (5)将 x 设为 y的左节点
        r.left = p;

        // (6)将 x的父节点 设为 y
        p.parent = r;
    }
}

右旋

右旋,就是以某个节点为支点向右旋转。

死磕 java集合之TreeMap源码分析(一)

整个右旋过程如下:

(1)将 x的右节点 设为 y的左节点,即 将 β 设为 y的左节点;

(2)将 y 设为 x的右节点的父节点,即 将 β的父节点 设为 y;

(3)将 y的父节点 设为 x的父节点;

(4)如果 y的父节点 是 空节点,则将x设为根节点;如果y是它父节点的左(右)节点,则将x设为y的父节点的左(右)节点;

(5)将 y 设为 x的右节点;

(6)将 y的父节点 设为 x;

让我们来看看treemap中的实现:

/**
 * 以p为支点进行右旋
 * 假设p为图中的y
 */
private void rotateright(entry<k,v> p) {
    if (p != null) {
        // p的左节点,即x
        entry<k,v> l = p.left;

        // (1)将 x的右节点 设为 y的左节点
        p.left = l.right;

        // (2)将 y 设为 x的右节点的父节点(如果x有右节点的话)
        if (l.right != null) l.right.parent = p;

        // (3)将 y的父节点 设为 x的父节点
        l.parent = p.parent;

        // (4)...
        if (p.parent == null)
            // 如果 y的父节点 是 空节点,则将x设为根节点
            root = l;
        else if (p.parent.right == p)
            // 如果y是它父节点的右节点,则将x设为y的父节点的右节点
            p.parent.right = l;
        else
            // 如果y是它父节点的左节点,则将x设为y的父节点的左节点
            p.parent.left = l;

        // (5)将 y 设为 x的右节点
        l.right = p;

        // (6)将 y的父节点 设为 x
        p.parent = l;
    }
}

未完待续,下一节我们一起探讨红黑树插入元素的操作。

现在公众号文章没办法留言了,如果有什么疑问或者建议请直接在公众号给我留言。


欢迎关注我的公众号“彤哥读源码”,查看更多源码系列文章, 与彤哥一起畅游源码的海洋。
死磕 java集合之TreeMap源码分析(一)