java 集合 5 - LinkedHashMap

参考文章:Java LinkedHashMap 源码解析,图解集合6:LinkedHashMap

基于 jdk 1.8 分析源码

LinkedHashMap允许一个null键和多个null值,有序,线程不安全。

LinkedHashMap直接使用父类HashMap的数组&链表结构保存数据,同时自己又采用了环型双向链表来维护元素顺序。

LinkedHashMap提供了两种形式的顺序:
访问顺序(access order):非常实用,可以使用这种顺序实现LRU(Least Recently Used)缓存(详见get方法解析)
插入顺序(insertion orde):元素的插入顺序(同一key的多次插入不会影响其顺序,详见newNode方法解析)

先来看看这两种顺序是的效果:

访问顺序

java 集合 5 - LinkedHashMap_第1张图片
这里写图片描述

可见通过get方法访问了Map中的元素,之后遍历值集合时遍历顺序受到了访问操作的影响,即“最近”被访问的元素排在值集合(即链表)的最前面。

这样在实现LRU算法时就能很方便的找到最近最少被使用的元素,即链表的头元素。

插入顺序

java 集合 5 - LinkedHashMap_第2张图片
这里写图片描述

默认采用的就是“插入顺序”。遍历时就会以插入顺序遍历。

Entry

    /**
     * HashMap.Node subclass for normal LinkedHashMap entries.
     */
    static class Entry extends HashMap.Node {
        Entry before, after;
        Entry(int hash, K key, V value, Node next) {
            super(hash, key, value, next);
        }
    }

LinkedHashMap直接使用了HashMap的数据结构来保存元素,不同的是相比HashMap的Entry增加了before和after成员,这两个成员用于维护元素的顺序。

构造器

    public LinkedHashMap() {
        super();
        accessOrder = false;
    }
    
    public LinkedHashMap(int initialCapacity,
                         float loadFactor,
                         boolean accessOrder) {
        super(initialCapacity, loadFactor);
        this.accessOrder = accessOrder;
    }

LinkedHashMap继承自HashMap,所以构造其中的super调用了父类HashMap的构造器。

要想使用“访问顺序”形式时只能通过调用三个参数的构造器创建实例。

put

LinkedHashMap并未重写HashMap的put方法,即直接调用了HashMap的put方法,也就是说LinkedHashMap插入元素时并未采取额外措施以实现其“有序”功能,那么该功能是在哪里实现的呢?

get

    public V get(Object key) {
        Node e;
        if ((e = getNode(hash(key), key)) == null)
            return null;
        if (accessOrder)
            afterNodeAccess(e);
        return e.value;
    }

get方法用于获得指定键对应的值(如果存在的话)。

注意if(accessOrder) afterNodeAccess(e);一句,上面提到LinkedHashMap的“有序”有两种体现形式,一种为“插入顺序”,一种为“访问顺序”,而accessOrder成员就是“访问顺序”形式的启用开关。

afterNodeAccess

    void afterNodeAccess(Node e) { // move node to last
        LinkedHashMap.Entry last;
        if (accessOrder && (last = tail) != e) {
            LinkedHashMap.Entry p =
                (LinkedHashMap.Entry)e, b = p.before, a = p.after;
            p.after = null;
            if (b == null) 
                head = a;
            else 
                b.after = a; 
            if (a != null) 
                a.before = b;
            else 
                last = b; 
            if (last == null)
                head = p;
            else {
                p.before = last;
                last.after = p;
            }
            tail = p;
            ++modCount;
        }
    }

如果开启了“访问顺序”的形式(创建实例时使用三个参数的构造器),并且当前访问的元素不是链表的尾结点,那么就会将当前访问结点变换为尾结点。

即最外层if条件成立,之后将当前访问结点强转为记录有前驱后继的LinkedHashMap专用结点,这里需要注意:参看上面的put方法我们知道put方法直接调用了HashMap的put方法,那么按理插入的结点类型也是HashMap.Node类型的,怎么能够强转为LinkedHashMap.Entry类型呢?

原因是这样的:参考HashMap源码可以知道,put方法内部调用了putVal,putVal中创建新节点调用的是newNode方法(而不是直接 new HashMap.Node),而LinkedHashMap重写了这个方法:

HashMap#newNode

    Node newNode(int hash, K key, V value, Node next) {
        return new Node<>(hash, key, value, next);
    }

LinkedHashMap#newNode

    Node newNode(int hash, K key, V value, Node e) {
        LinkedHashMap.Entry p =
            new LinkedHashMap.Entry(hash, key, value, e);
        linkNodeLast(p);
        return p;
    }

所以这里强转就不会出现ClassCastException

之后的步骤就是将当前访问结点变换为尾结点的过程了(有点复杂,没理清)。

newNode

    Node newNode(int hash, K key, V value, Node e) {
        LinkedHashMap.Entry p =
            new LinkedHashMap.Entry(hash, key, value, e);
        linkNodeLast(p);
        return p;
    }

上面已经说了一下newNode方法的,除了用于创建新元素以外,在LinkedHashMap中,newNode方法还额外做了一件事情:将当前元素链接到链表尾,即创建LinkedHashMap专用的LinkedHashMap.Entry,构造该节点的前驱和后继以维护其“插入顺序”。对应linkNodeLast方法。

 private void linkNodeLast(LinkedHashMap.Entry p) {
        LinkedHashMap.Entry last = tail;
        tail = p;
        if (last == null)
            head = p;
        else {
            p.before = last;
            last.after = p;
        }
    }

如果尾结点为空,这意味着LinkedHashMap中没有元素,那么新元素就是链表头(head=p),同时链表尾元素也指向它(tail=p);否则将链表尾元素替换为新的元素(p.before=last;last.after=p;)。

你可能感兴趣的:(java 集合 5 - LinkedHashMap)