从LinkedHashMap的常用方法来解析它的内部实现

上两篇文章
http://www.jianshu.com/p/a122c79ee60c
http://www.jianshu.com/p/336b1b45d140
我们讲了HashMap的实现原理,今天我们来讲讲HashMap的子类LinkedHashMap的内部实现。

构造方法

因为它是HashMap的子类,所以他们的存储都是数组形式,不同的是它添加了一个排序的参数accessOrder,这个参数对后面的方法实现都会有影响,可以通过构造方法给它赋值


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

initialCapacity默认数组的大小;loadFactor是加载因子,默认是0.75,我们添加数据的数量大于这个加载因子loadFactor时就会扩充数组的大小;accessOrder就是排序。

put方法

因为它是HashMap的子类,所以它的put方法和HashMap的put方法一样,唯一不同的是它重写了put方法当中的(put方法的解析可以看我文章开头链接)addNewEntry方法和preModify方法,我们先来看看addNewEntry方法

    @Override 
    void addNewEntry(K key, V value, int hash, int index) {
        //LinkedHashMap内部维护了一个双向链表, header是链表的头
        LinkedEntry<K, V> header = this.header;

        // 如果LinkedHashMap有数据,并且removeEldestEntry方法返回true,
         // 就移除掉最早添加的数据, removeEldestEntry默认返回false
        LinkedEntry<K, V> eldest = header.nxt;
        if (eldest != header && removeEldestEntry(eldest)) {
            remove(eldest.key);
        }

        // new一个LinkedEntry对象,最后面的两个参数一个是这个对象的上一个属性值,
          //  一个是下一个属性值,table[index]对象是维护它在数组当中的单向链表值
        LinkedEntry<K, V> oldTail = header.prv;
        LinkedEntry<K, V> newTail = new LinkedEntry<K,V>(
                key, value, hash, table[index], header, oldTail);
        table[index] = oldTail.nxt = header.prv = newTail;
    }

从上面方法可以得出结论:LinkedHashMap维持了HashMap原来的数据结构,并且添加了一个双向链表,所以LinkedHashMap是有序的,在put数据的同时可以通过removeEldestEntry方法来决定是否删除最早添加的数据。

我们再来看看preModify方法,preModify方法是在有重复key的时候会调用

@Override 
void preModify(HashMapEntry<K, V> e) {
        //如果为true就执行makeTail方法
        if (accessOrder) {
            makeTail((LinkedEntry<K, V>) e);
        }
}

private void makeTail(LinkedEntry<K, V> e) {
     // e是那个重复key的 对象
     //就是把链表当中的e对象删除掉,然后让e的上一个和下一个对象通过链表的形式关联起来
     e.prv.nxt = e.nxt;
     e.nxt.prv = e.prv;

     // 然后把e添加到最后
     LinkedEntry<K, V> header = this.header;
     LinkedEntry<K, V> oldTail = header.prv;
     e.nxt = header;
     e.prv = oldTail;
     oldTail.nxt = header.prv = e;
     modCount++;
}

从上面的方法可以得出结论:如果有重复的key并且accessOrder是true,就会把重复key的对象移动到最后;put方法到此就解析完了,下面来看看get方法

    @Override 
public V get(Object key) {
        /*
         * This method is overridden to eliminate the need for a polymorphic
         * invocation in superclass at the expense of code duplication.
         */
        if (key == null) {
            HashMapEntry<K, V> e = entryForNullKey;
            if (e == null)
                return null;
            if (accessOrder)
                makeTail((LinkedEntry<K, V>) e);
            return e.value;
        }

        int hash = Collections.secondaryHash(key);
        HashMapEntry<K, V>[] tab = table;
        for (HashMapEntry<K, V> e = tab[hash & (tab.length - 1)];
                e != null; e = e.next) {
            K eKey = e.key;
            if (eKey == key || (e.hash == hash && key.equals(eKey))) {
                if (accessOrder)
                    makeTail((LinkedEntry<K, V>) e);
                return e.value;
            }
        }
        return null;
    }

上面的逻辑很容易理解,就是说如果accessOrder为true就执行makeTail方法,makeTail方法上面已经分析过了,从上面代码可以得出结论:如果在我取值的时候accessOrder为true,那么就会把我取的这个值移动到链表的最后,header的上面。

代码验证
首先来验证一下遇到重复key,accessOrder不同值的情况
 LinkedHashMap linkedHashMap = new LinkedHashMap(16 , 0.75f, false);
 linkedHashMap.put("1", "one");
 linkedHashMap.put("2", "two");
 linkedHashMap.put("3", "three");
 linkedHashMap.put("4", "four");
 linkedHashMap.put("1", "one");

Iterator iter = linkedHashMap.entrySet().iterator();
while (iter.hasNext()) {
       Map.Entry entry = (Map.Entry) iter.next();
       System.out.println("IndexActivity Value:" + entry.getValue());
}

首先设置accessOrder为false,看看打印结果

IndexActivity Value:one
IndexActivity Value:two
IndexActivity Value:three
IndexActivity Value:four

因为accessOrder为false,所以在key重复的时候只会修改它的value,并不会移动它的位置,那设置为true会是什么结果呢?我们来看看打印结果

IndexActivity Value:two
IndexActivity Value:three
IndexActivity Value:four
IndexActivity Value:one

因为key=1重复并且accessOrder为true,所以会把key=1的这个对象移动到链表的最后。

下面来验证一下accessOrder不同值get操作的情况
  LinkedHashMap linkedHashMap = new LinkedHashMap(16 , 0.75f, false);
        linkedHashMap.put("1", "one");
        linkedHashMap.put("2", "two");
        linkedHashMap.put("3", "three");
        linkedHashMap.put("4", "four");

        linkedHashMap.get("3");
        Iterator iter = linkedHashMap.entrySet().iterator();
        while (iter.hasNext()) {
            Map.Entry entry = (Map.Entry) iter.next();
            System.out.println("IndexActivity Value:" + entry.getValue());
  }

首先设置accessOrder为false,看看打印结果

IndexActivity Value:one
IndexActivity Value:two
IndexActivity Value:three
IndexActivity Value:four

还是原来的顺序,下面我们设置accessOrder为true,再看看打印结果

IndexActivity Value:one
IndexActivity Value:two
IndexActivity Value:four
IndexActivity Value:three

可以看到three被移动到了最后面。

remove方法

remove方法和hashMap的remove方法一样(remove方法的解析可以查看我文章开头的链接),区别在于重写了remove方法当中的postRemove方法

 @Override 
 void postRemove(HashMapEntry<K, V> e) {
        LinkedEntry<K, V> le = (LinkedEntry<K, V>) e;
        //把e对象的上个对象和下个对象用链表的形式关联起来
        le.prv.nxt = le.nxt;
        le.nxt.prv = le.prv;
        //把e对象指向上个对象和下个对象的属性设置为null
        le.nxt = le.prv = null; // Help the GC (for performance)
    }

到此LinkedHashMap的常用方法就分析完成了,其他的方法大家可以去看源码,都很容易理解的。

总结

从上面的方法可以看出:LinkedHashMap和HashMap区别就是多了个双向链表,它是有序的,可以通过get方法改变它的顺序(前提是accessOrder=true),在put数据的同时通过removeEldestEntry方法返回的值来决定是否移除数组第一位的数据。

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

推荐阅读更多精彩内容

  • 一、基本数据类型 注释 单行注释:// 区域注释:/* */ 文档注释:/** */ 数值 对于byte类型而言...
    龙猫小爷阅读 4,258评论 0 16
  • Collection & Map Collection 子类有 List 和 Set List --> Array...
    任教主来也阅读 3,161评论 1 9
  • 实际上,HashSet 和 HashMap 之间有很多相似之处,对于 HashSet 而言,系统采用 Hash 算...
    曹振华阅读 2,511评论 1 37
  • LinkedHashMap 概述 HashMap 是无序的,HashMap 在 put 的时候是根据 key 的 ...
    曹振华阅读 826评论 0 6
  • 狂风骤雨一般的一边是火焰一边是冰块的永不停歇的像个孩子一样高兴的像个老头子一样顽固的又像个女孩子一样突然就要生气的...
    艾黑丫阅读 318评论 11 26