LinkedHashMap继承自HashMap,除了提供HashMap的功能外,LinkedHashMap还是维护一个双向链表(实际为带头结点的双向循环链表),持有所有的键值对的引用:

  1. 这个双向链表定义了迭代器的迭代顺序,默认按插入顺序迭代;
  2. 也可以在构造时设置为按照LRU方式(访问顺序)迭代(from least-recently accessed to most-recently access-order),最近最少访问的键值对放在链表最前(头结点之后的第一个结点);

废话不多说,直接看源码:

// @since 1.4
public class LinkedHashMap<K,V> extends HashMap<K,V> implements Map<K,V>{
private static final long serialVersionUID = 3801124242820219131L; // 双向循环链表的头结点
private transient Entry<K,V> header; // 迭代的顺序,按访问顺序(access-order)时为true,按插入顺序(insertion-order)时为false
private final boolean accessOrder; public LinkedHashMap(int initialCapacity,float loadFactor){
super(initialCapacity,loadFactor);
// 默认为按插入顺序
accessOrder = false;
} public LinkedHashMap(int initialCapacity,float loadFactor,boolean accessOrder){
super(initialCapacity,loadFactor);
this.accessOrder = accessOrder;
}
// 各种构造方法,省略... // 构造器和伪构造器(包括clone(),readObject())中会调用该方法
void init(){
// 头结点
header = new Entry<>(-1,null,null,null);
header.before = header.after = header;
} // 扩容时会调用该方法
void transfer(HashMap.Entry[] newTable,boolean rehash){
int newCapacity = newTable.length;
// 这里通过链表遍历,重新计算每个的hash值,而不是遍历hash表;
// 在扩容时,链表并没有发生变化;
for(Entry<K,V> e = header.after;e != header;e = e.after){
if(rehash){
e.hash = (e.key == null) ? 0 : hash(e.key);
}
int index = indexFor(e.hash,newCapacity);
e.next = newTable[index];
newTable[index] = e;
}
} // 通过链表来查找是否存在指定元素,这样更快一点
public boolean containsValue(Object value){
if(value == null){
for(Entry e = header.after;e != header;e = e.after)
if(e.value == null){
return true;
}
}else{
for(Entry e = header.after;e != header;e = e.after)
if(value.equals(e.value)){
return true;
}
}
return false;
} public V get(Object key){
Entry<K,V> e = (Entry<K,V>)getEntry(key);
if(e == null){
return null;
}
// 被访问的entry,在链表中排序
e.recordAccess(this);
return e.value;
} // 清除所有元素时,也要将所有的entry从链表中删除,不再持有这些entry的引用
public void clear(){
super.clear();
header.before = header.after = header;
} 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;
} // 在双向链表中指定结点之前插入当前结点
private void addBefore(Entry<K,V> existingEntry){
after = existingEntry;
before = existingEntry.before;
before.after = this;
after.before = this;
} // 当map中的一个已有的entry的值被访问(get())或修改(put()修改已有key的value)时被调用
void recordAccess(HashMap<K,V> m){
LinkedHashMap<K,V> lm = (LinkedHashMap<K,V>)m;
// 如果LinkedHashMao是按访问顺序,这里会将被修改的entry移到链表最后面(头结点的前面);
// 否则什么都不干;
// 也就是最近被访问的,放在链表的最后;最近最少访问的(LRU),放在链表最前;
if(lm.accessOrder){
lm.modCount++;
remove();
addBefore(lm.header);
}
} // 当前entry被从map中移除时被调用,在这里将当前结点从双向链表中移除
void recordRemoval(HashMap<K,V> m){
remove();
}
} private abstract class LinkedHashIterator<T> implements Iterator<T>{
Entry<K,V> nextEntry = header.after;
Entry<K,V> lastReturned = null;
int expectedModCount = modCount; public boolean hasNext(){
return nextEntry != header;
}
public void remove(){
if(lastReturned == null){
throw new IllegalStateException();
}
if(modCount != expectedModCount){
throw new ConcurrentModificationException();
}
LinkedHashMap.this.remove(lastReturned.key);
lastReturned = null;
expectedModCount = modCount;
} // 通过链表顺序迭代
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;
}
} private class KeyIterator extends LinkedHashIterator<K>{
public K next(){ return nextEntry().getKey(); }
}
private class ValueIterator extends LinkedHashIterator<V>{
public V next(){ return nextEntry().value; }
}
private class EntryIterator extends LinkedHashIterator<Map.Entry<K,V>>{
public Map.Entry<K,V> next(){ return nextEntry(); }
} Iterator<K> newKeyIterator(){ return new KeyIterator(); }
Iterator<V> newValueIterator(){ return new ValueIterator(); }
Iterator<Map.Entry<K,V>> newEntryIterator(){ return new EntryIterator(); } // put()会调用该方法
void addEntry(int hash,K key,V value,int bucketIndex){
super.addEntry(hash,key,value,bucketIndex);
Entry<K,V> eldest = header.after;
// 判断是否移除"最老"的元素;可以通过修改removeEldestEntry()的返回值来改变这一特性
// 默认不移除
if(removeEldestEntry(eldest)){
removeEntryForKey(eldest.key);
}
} // put()中的addEntry()会调用该方法,将添加的entry放到链表尾
// 不管是按什么顺序(accessOrder),添加一个新entry时,都会将添加的entry放到链表尾,
// 因为新添加的entry即是最后插入的,也是是最近访问的
void createEntry(int hash,K key,V value,int bucketIndex){
HashMap.Entry<K,V> old = table[bucketIndex];
Entry<K,V> e = new Entry<>(hash,key,value,old);
table[bucketIndex] = e;
// 添加到链表尾
e.addBefore(header);
size++;
} // 如果希望每次put()时,移除"最老"的结点,返回true,默认返回false
// "最老"的结点:如果为访问顺序,则为最久没被访问的结点;如果为插入顺序,则为最早被添加的元素;
protected boolean removeEldestEntry(Map.Entry<K,V> eldest){
return false;
} }

有上述源码可以看出,LinkedHashMapHashMap的基础上,添加了如下特点:

  1. 插入的元素保持有序,使得遍历时是有序的;且这种顺序可以按照需要,配置为如下2种之一:
  • 访问顺序:最近最少访问的在最前,最近访问的元素在最后;

  • 插入顺序:最早插入的元素在最前,最后添加的元素在最后;

    默认为按插入顺序;

  1. 因为可以按照访问顺序有序,所以可以用来支持LRU算法;
  2. 可配置删除最老的结点,在每次添加结点的时候,判断是否将链表最前的结点删除;

Java容器解析系列(12) LinkedHashMap 详解的更多相关文章

  1. Java容器解析系列(11) HashMap 详解

    本篇我们来介绍一个最常用的Map结构--HashMap 关于HashMap,关于其基本原理,网上对其进行讲解的博客非常多,且很多都写的比较好,所以.... 这里直接贴上地址: 关于hash算法: Ha ...

  2. Java容器解析系列(13) WeakHashMap详解

    关于WeakHashMap其实没有太多可说的,其与HashMap大致相同,区别就在于: 对每个key的引用方式为弱引用; 关于java4种引用方式,参考java Reference 网上很多说 弱引用 ...

  3. Java容器解析系列(7) ArrayDeque 详解

    ArrayDeque,从名字上就可以看出来,其是通过数组实现的双端队列,我们先来看其源码: /** 有自动扩容机制; 不是线程安全的; 不允许添加null; 作为栈使用时比java.util.Stac ...

  4. Java容器解析系列(9) PrioriyQueue详解

    PriorityQueue:优先级队列; 在介绍该类之前,我们需要先了解一种数据结构--堆,在有些书上也直接称之为优先队列: 堆(Heap)是是具有下列性质的完全二叉树:每个结点的值都 >= 其 ...

  5. Java容器解析系列(17) LruCache详解

    在之前讲LinkedHashMap的时候,我们说起可以用来实现LRU(least recent used)算法,接下来我看一下其中的一个具体实现-----android sdk 中的LruCache. ...

  6. Java容器解析系列(14) IdentityHashMap详解

    IdentityHashMap,使用什么的跟HashMap相同,主要不同点在于: 数据结构:使用一个数组table来存储 key:value,table[2k] 为key, table[2k + 1] ...

  7. Java容器解析系列(0) 开篇

    最近刚好学习完成数据结构与算法相关内容: Data-Structures-and-Algorithm-Analysis 想结合Java中的容器类加深一下理解,因为之前对Java的容器类理解不是很深刻, ...

  8. java基础解析系列(四)---LinkedHashMap的原理及LRU算法的实现

    java基础解析系列(四)---LinkedHashMap的原理及LRU算法的实现 java基础解析系列(一)---String.StringBuffer.StringBuilder java基础解析 ...

  9. Java容器解析系列(10) Map AbstractMap 详解

    前面介绍了List和Queue相关源码,这篇开始,我们先来学习一种java集合中的除Collection外的另一个分支------Map,这一分支的类图结构如下: 这里为什么不先介绍Set相关:因为很 ...

随机推荐

  1. Vue学习笔记-使用ElementUI

    ElementUI官方地址:https://element.eleme.cn/2.11/#/zh-CN 1.初期准备 首先我们准备几个基本的样式文件:normalize.css 和 base.css ...

  2. Java 学习笔记(9)——java常用类

    之前将Java的大部分语法都回顾完了,后面添加一些常见的操作,基础语法就结束了.至于在这里再次提到常用类是由于有一部分体现在使用它的继承类或者接口之类的.这些需要有面向对象编程的基础 Object类 ...

  3. tag的使用

    tag = True while tag: print("level") choice = input("level>>>").strip() ...

  4. 大学最新毕业论文参考文献,包含java,jsp,mysql,Android,sql,PHP

    每当毕业论文写到最后需要参考文献时,往往是很令人头疼的,因为有的老师对参考文献的要求是很多的,比如需要国内的和国外的,时间必须是近三年的,满足XXX要求的文献至少需要三篇以上等等.今天我就来给大家整理 ...

  5. 多vps管理面板

           iis7远程桌面连接工具,又叫做iis7远程桌面管理软件,是一款绿色小巧,功能实用的远程桌面管理工具,其界面简洁,操作便捷,能够同时远程操作多台服务器,并且多台服务器间可以自由切换,适用 ...

  6. 重新精读《Java 编程思想》系列之final关键字

    在java中final关键字标识无法被修改.接下来从final修饰数据.方法和类进行介绍. final数据 final用来告知编译器这一块数据是恒定不变的.数据恒定不变又如下作用: 1.一个永不改变的 ...

  7. Spring的BeanPostProcessor后置处理器与bean的生命周期

    前言 本文将把Spring在Bean的生命周期中涉及到的后置处理器一一梳理出来,并简要说一下功能,至于每个后置处理器在实际扩展中的用处,还要后续慢慢探索总结. 正文 下面一步步跟进探寻那些后置处理器们 ...

  8. 你好,babel

    写在前面 其实学babel是本人2019年Q3的一个计划,因为当时自己做的一个项目需要自己去配babel,也遇到了一些困难,发现自己对babel的了解还是很少的,所以决定好好看下babel:可是后来解 ...

  9. 1078 字符串压缩与解压 (20分)C语言

    文本压缩有很多种方法,这里我们只考虑最简单的一种:把由相同字符组成的一个连续的片段用这个字符和片段中含有这个字符的个数来表示.例如 ccccc 就用 5c 来表示.如果字符没有重复,就原样输出.例如 ...

  10. 1088 三人行 (20分)C语言

    子曰:"三人行,必有我师焉.择其善者而从之,其不善者而改之." 本题给定甲.乙.丙三个人的能力值关系为:甲的能力值确定是 2 位正整数:把甲的能力值的 2 个数字调换位置就是乙的能 ...