java.util.LinkedHashMap双向链表有序源码(jdk1.7)

准备工作

由于LinkedHashMap也是继承HashMap,在HashMap类的基础上进行的功能扩展,所以先了解下HashMap :https://www.jianshu.com/p/374546518bb6

LinkedHashMap中链地址的双向循环链表结构

        // 双向循环链表维护冲突值 
        private static class Entry<K,V> extends HashMap.Entry<K,V> {  
            // 双向循环链表的前驱节点和后继结点  
            Entry<K,V> before, after;  
      
            Entry(int hash, K key, V value, HashMap.Entry<K,V> next) {  
                super(hash, key, value, next);  
        }  
        // 通过前驱后继关系将节点删除    
        private void remove() {  
            // 当前节点的前驱节点的后继为当前节点的后继结点  
            before.after = after;  
            // 当前节点的后继结点的前驱为当前节点的前驱节点  
            after.before = before;  
        }  

说明:使用前驱节点和后继节点删除当前节点

        /**  
         * 将existingEntry指定节点前面插入当前节点。  
         * existingEntry :现有项  
         */  
        private void addBefore(Entry<K,V> existingEntry) {  
            //this.after =existingEntry;  
            after  = existingEntry;  
            //this.before =existingEntry.before;  
            before = existingEntry.before;  
            before.after = this;  
            after.before = this;  
        }  

说明(重要设计):existingEntry指定节点作为当前节点的后继节点,existingEntry指定节点的前驱节点作为当前节点的前驱节点,对于before和after本身就是当前节点的前驱节点和后继节点,再修改下,让前驱节点的后继指针指向当前节点,让后继节点的前驱指针指向当前节点即可。完成了将existingEntry指定节点前面插入当前节点。

        /**  
         *  如果LinkedHashMap的排序顺序为访问顺序,那么就将当前节点插入到头结点前面  
         */  
        void recordAccess(HashMap<K,V> m) {  
            LinkedHashMap<K,V> lm = (LinkedHashMap<K,V>)m;  
            //如果排序顺序为访问顺序  
            if (lm.accessOrder) {  
                // LinkedHashMap修改次数加1
                lm.modCount++;  
                // 删除当前节点  
                remove();  
                // 在头节点前面插入当前节点  
                addBefore(lm.header);  
            }  
        }  

说明:如果排序顺序为访问顺序,那么将在头结点前面插入当前节点。

        void recordRemoval(HashMap<K,V> m) {  
            remove();  
        }  
    }  

属性

    // 双向链表的头结点    
    private transient Entry<K,V> header;  

说明:双向链表的头节点。

    // 排序模式:对于访问顺序,为 true;对于插入顺序,则为 false。   
    private final boolean accessOrder; 

说明:节点的排序方式:对于访问顺序,为 true;对于插入顺序,则为 false。

构造方法

构造方法1:传入初始化容量,加载因子,默认采用插入顺序排序的HashMap构造

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

构造方法2:传入初始化容量,默认加载因子为0.75,默认采用插入顺序排序的HashMap构造

    public LinkedHashMap(int initialCapacity) {  
        super(initialCapacity);  
        accessOrder = false;  
    }  

构造方法3:采用默认初始化容量16,默认加载因子为0.75,默认插入顺序排序的HashMap构造

    public LinkedHashMap() {  
        super();  
        accessOrder = false;  
    }  

构造方法4:构造一个映射关系与指定 Map 相同的 HashMap; 所创建的 HashMap 具有默认的加载因子 (0.75) 和足以容纳指定 Map 中映射关系的初始容量; 采用默认插入顺序排序

    public LinkedHashMap(Map<? extends K, ? extends V> m) {  
        super(m);  
        accessOrder = false;  
    }  

构造方法5:构造一个拥有初始化容量、加载因子、排序顺序的LinkedHashMap

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

方法

init方法,初始化双向循环链表

   @Override  
    void init() {  
        header = new Entry<>(-1, null, null, null);  
        header.before = header.after = header;  
    }  

createEntry方法:创建节点,将头结点插入到当前节点的前面。

void createEntry(int hash, K key, V value, int bucketIndex) { 
    // 通过下标获取键表中对应的节点Entry
    HashMap.Entry<K,V> old = table[bucketIndex];
    // 创建一个节点,后继节点指向old
    Entry<K,V> e = new Entry<>(hash, key, value, old);  
    // 插入节点e
    table[bucketIndex] = e;  
    // e节点设置为头结点的后继节点
    e.addBefore(header);
    size++;  
}

说明:

  1. 虽然HashMap查找元素的方式是,通过键找到hash值,通过hash值找到下标,通过下标找到节点链。
  2. LinkedHashMap在原来的HashMap基础上,将单向节点链中的每个节点又额外构造了一条双向链表,插入值的顺序,就是在头结点的后面插入新节点(类似于不停的插队的节奏)。

transfer方法:将所有元素都放到新的数组中,重写父类的HashMap
的transfer方法。

        /**  
         * 将所有的元素放置到新的数组中  
         * 重写父类HashMap的transfer方法  
         */  
        @Override  
        void transfer(HashMap.Entry[] newTable, boolean rehash) {  
            int newCapacity = newTable.length;  
            // 使用双向链表的头结点进行循环遍历  
            for (Entry<K,V> e = header.after; e != header; e = e.after) {  
                if (rehash)  
                    e.hash = (e.key == null) ? 0 : hash(e.key);  
                // 通过hash值获取对应的下标索引  
                int index = indexFor(e.hash, newCapacity);  
                // 插入到链表表头  
                e.next = newTable[index];  
                // 新数组的索引对应的值为header  
                newTable[index] = e;  
            }  
        }  

说明:遍历双向链表,从头结点的后继节点开始遍历,然后将遍历到的节点设置到新的键表中。


get方法:返回此映射中映射到指定键的值。

    public V get(Object key) {  
            // 调用父类的getEntry方法,通过键key来获取对应的Entry  
            Entry<K,V> e = (Entry<K,V>)getEntry(key);  
             // 如果e为null,那么根本不存在对应的Entry就更没有对应的值了,所以返回null  
            if (e == null)  
                return null;  
            // 将当前节点插入到头结点前面  
            e.recordAccess(this);  
            return e.value;  
        }  

说明:通过键获取节点,通过节点获取值。


迭代:

private class KeyIterator extends LinkedHashIterator<K> {
     public K next() { return nextEntry().getKey(); }
}

Entry<K,V> nextEntry() {
     if (modCount != expectedModCount)
             throw new ConcurrentModificationException();
     if (nextEntry == header)
             throw new NoSuchElementException();

    Entry<K,V> e = lastReturned = nextEntry;
    nextEntry = e.after;
    return e;
 }

总结:

  1. LinkedHashMap存储冲突值使用双向循环链表。
  2. LinkedHashMap中键和值都允许存入null值。
  3. LinkedHashMap中值允许重复,如果发现键相同,就更新原来的值。
  4. LinkedHashMap是线程不安全的。
  5. LinkedHashMap中accessOrder属性,true意味着排序模式为访问顺序遍历出来,false意味着排序模式为插入顺序遍历出来。
  6. LinkedHashMap单独维护了一条双向链表,保证了按照插入的顺序设计的,所以取得时候就会保证有序。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,080评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,422评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,630评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,554评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,662评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,856评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,014评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,752评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,212评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,541评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,687评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,347评论 4 331
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,973评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,777评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,006评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,406评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,576评论 2 349

推荐阅读更多精彩内容