概述

文章的内容基于JDK1.7进行分析,之所以选用这个版本,是因为1.8的有些类做了改动,增加了阅读的难度,虽然是1.7,但是对于1.8做了重大改动的内容,文章也会进行说明。

LinkedHashMap,见名知义,带链表的HashMap, 所以LinkedHashMap是有序,LinkedHashMap作为HashMap的扩展,它改变了HashMap无序的特征。它使用了一个双向的链表来会维护key-value对的次序,该链表维护了map的迭代顺序,该迭代顺序和key-value对的插入顺序保持一致。LinkedHashMap重写了父类的一些方法,这些方法也会在下面的文章中进行说明。

数据结构继承关系java.lang.Object

java.util.AbstractMap

java.util.HashMap

java.util.LinkedHashMap

从上面的集成关系中看出,LinkedHashMap集成了HashMap类,所以便拥有了HashMap开放的所有功能,而LinkedList在所有功能的基础上又进行了升级,添加了记住元素添加顺序的职责。实现接口Serializable, Cloneable, Map

LinkedHashMap 可序列化,可以被克隆 ,实现了Map接口基本属性private transient Entry header;  //双向链表的头节点private final boolean accessOrder;  //排序的规则,false按插入顺序排序,true访问顺序排序

重要方法深度解析构造方法//LinkedHashMap的构造方法,都是通过调用父类的构造方法来实现,大部分accessOrder默认为falsepublic LinkedHashMap(int initialCapacity, float loadFactor) {    super(initialCapacity, loadFactor);

accessOrder = false;

}public LinkedHashMap(int initialCapacity) {    super(initialCapacity);

accessOrder = false;

}public LinkedHashMap() {    super();

accessOrder = false;

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

accessOrder = false;

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

}

上面是LinkedHashMap的构造方法,通过传入初始化参数和代码看出,LinkedHashMap的构造方法和父类的构造方法,是一一对应的。也是通过super()关键字来调用父类的构造方法来进行初始化,唯一的不同是最后一个构造方法,提供了AccessOrder参数,用来指定LinkedHashMap的排序方式,accessOrder =false -> 插入顺序进行排序 , accessOrder = true -> 访问顺序进行排序。

put()方法

LinkedHashMap并没有重写父类的put()方法,说明调用put方法时实际上调用的是父类的put方法。

get()方法public V get(Object key) {

Entry e = (Entry)getEntry(key); //调用父类的getEntry()方法

if (e == null)        return null;

e.recordAccess(this);  //判断排序方式,如果accessOrder = true , 删除当前e节点

return e.value;

}

remove()

LinkedHashMap并没有重写父类的remove()方法,说明调用remove方法时实际上调用的是父类的remove()方法。

源码解析Entry定义private static class Entry extends HashMap.Entry {    //定义Entry类型的两个变量,或者称之为前后的两个指针

Entry before, after;    //构造方法与HashMap的没有区别,也是调用父类的Entry构造方法

Entry(int hash, K key, V value, HashMap.Entry next) {            super(hash, key, value, next);

}    //删除

private void remove() {

before.after = after;

after.before = before;

}    //插入节点到指定的节点之前

private void addBefore(Entry existingEntry) {

after  = existingEntry;

before = existingEntry.before;

before.after = this;

after.before = this;

}    //方法重写,HashMap中为空

void recordAccess(HashMap m) {

LinkedHashMap lm = (LinkedHashMap)m;        if (lm.accessOrder) {

lm.modCount++;

remove();

addBefore(lm.header);

}

}    //方法重写 ,HashMap中方法为空

void recordRemoval(HashMap m) {

remove();

}

}public class LinkedHashMap extends HashMap    implements Map{    //可序列化版本号

private static final long serialVersionUID = 3801124242820219131L;    //双向链表的头指针

private transient Entry header;    //双向链表的排序方法,false 插入顺序排序,true访问顺序排序

private final boolean accessOrder;    //构造方法,指定初始大小,指定负载因子

public LinkedHashMap(int initialCapacity, float loadFactor) {        super(initialCapacity, loadFactor);

accessOrder = false;

}    //构造方法,指定初始容量

public LinkedHashMap(int initialCapacity) {        super(initialCapacity);

accessOrder = false;

}    //无参构造方法k,采用默认参数

public LinkedHashMap() {        super();

accessOrder = false;

}    //将指定的m集合转化为LinkedHashmap存储

public LinkedHashMap(Map extends K, ? extends V> m) {        super(m);

accessOrder = false;

}    //构造方法,指定初始容量,负载因子和排序方式

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

}    //重写父类init方法,init方法在父类构造函数被调用,初始化双向链表

//header的前驱和后继都是指向它自己

@Override

void init() {

header = new Entry<>(-1, null, null, null);

header.before = header.after = header;

}    //重写父类的transfer方法,在HashMap执行扩容操作时被调用,HashMap中的是通过遍历Entry[]数组的方式来实现数据的拷贝复制,重写后是通过遍历双向链表的方式来进行数据的复制。遍历双向链表的方式效率上更高一些

@Override

void transfer(HashMap.Entry[] newTable, boolean rehash) {        int newCapacity = newTable.length;        for (Entry 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;

}

}    //重写父类的containsValue, 由遍历数组的方式修改为遍历列表的方式

public boolean containsValue(Object value) {        // Overridden to take advantage of faster iterator

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;

}    //重写父类的get方法

public V get(Object key) {

Entry e = (Entry)getEntry(key); //返回实体

if (e == null)            return null;

e.recordAccess(this); //如果是访问顺序排序,则将e移动到链表的末尾处

return e.value;

}    //清除集合

public void clear() {        super.clear();

header.before = header.after = header;

}    //内部类实现了迭代方法

private abstract class LinkedHashIterator implements Iterator {

Entry nextEntry    = header.after;

Entry 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 nextEntry() {            if (modCount != expectedModCount)                throw new ConcurrentModificationException();            if (nextEntry == header)                throw new NoSuchElementException();

Entry e = lastReturned = nextEntry;

nextEntry = e.after;            return e;

}

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

}    private class ValueIterator extends LinkedHashIterator {        public V next() { return nextEntry().value; }

}    private class EntryIterator extends LinkedHashIterator> {        public Map.Entry next() { return nextEntry(); }

}    // These Overrides alter the behavior of superclass view iterator() methods

Iterator newKeyIterator()   { return new KeyIterator();   }    Iterator newValueIterator() { return new ValueIterator(); }

Iterator> newEntryIterator() { return new EntryIterator(); }    //重写父类的addEntry方法

void addEntry(int hash, K key, V value, int bucketIndex) {        super.addEntry(hash, key, value, bucketIndex);

Entry eldest = header.after;        if (removeEldestEntry(eldest)) {

removeEntryForKey(eldest.key);

}

}    //重写createEntry方法

//执行两步操作:

// 1. 添加到table数组中, 2 . 插入到双向链表中

void createEntry(int hash, K key, V value, int bucketIndex) {

HashMap.Entry old = table[bucketIndex];

Entry e = new Entry<>(hash, key, value, old);

table[bucketIndex] = e;

e.addBefore(header);

size++;

}    protected boolean removeEldestEntry(Map.Entry eldest) {        return false;

}

}

下面详细的分析一下LinkedHashMap中操作的实现:HashMap h = new LinkedHashMap();

h.put("张三", 18);

上面是简短的两句代码,来看一下到底包含了何种的操作。

图解LinkedHashMap的put操作.png

上面的图片已经描述的很清楚了,在此不再添加文字的表述。

关于删除方法,感兴趣的可以自行研究。

总结

LinkedHashMap继承了HashMap类,重写了部分方法,在HashMap中一些空的实现,LinkedHashMap都做了实现,扩展了HashMap类的功能,LinkedHashMap可以保存元素的插入顺序,顺序有两种方式一种是按照插入顺序排序,一种按照访问做排序。默认以插入顺序排序,性能比HashMap略低,线程也是不安全的。

至此,Java集合的源码系列就分析完了,像HashTable,stack等集合从开发者的角度上已经不建议在使用,因为已经是比较古老的类,有了更好类做了替代。

作者:起个名忒难

链接:https://www.jianshu.com/p/daf16e703049

android map 底层实现原理,LinkedHashMap底层实现和原理(源码解析)相关推荐

  1. Android技术栈(五)HashMap(包括红黑树)与ArrayMap源码解析

    1 总览 本文会对 Android 中常用HashMap(有红黑树)和ArrayMap进行源码解析,其中 HashMap 源码来自 Android Framework API 28 (JDK=1.8) ...

  2. android视频缓存框架 [AndroidVideoCache](https://github.com/danikula/AndroidVideoCache) 源码解析与评估

    文章目录 android视频缓存框架 [AndroidVideoCache](https://github.com/danikula/AndroidVideoCache) 源码解析与评估 引言 使用方 ...

  3. 云原生服务网格Istio:原理、实践、架构与源码解析

    华为云原生团队600多页的Istio实战精华总结,云原生服务网格Istio:原理.实践.架构与源码解析的电子书. 图书介绍 <云原生服务网格Istio:原理.实践.架构与源码解析>分为原理 ...

  4. android 输入法如何启动流程_android输入法02:openwnn源码解析01—输入流程

    android 输入法 02:openwnn 源码解析 01-输入流程 之后要开始 android 日文输入法的测试,因此现在开始研究 android 输入法.之前两 篇文章已经对 android 自 ...

  5. Android View系列(二):事件分发机制源码解析

    概述 在介绍点击事件规则之前,我们需要知道我们分析的是MotionEvent,即点击事件,所谓的事件分发就是对MotionEvent事件的分发过程,即当一个MotionEvent生成以后,系统需要把这 ...

  6. Android Mms短信的发送流程,短信发送源码解析

    发送前的校验 从短信的点击按钮开始着手: // packages/apps/Mms/src/com/android/mms/ui/ComposeMessageActivity.java@Overrid ...

  7. aop实现原理 - JDK动态代理(实例+源码解析)

    动态代理: jdk代理-基于接口代理 通过 类:java.lang.reflect.Proxy 生成动态代理类 实现 接口:InvocationHandler 只能基于接口进行动态代理 代码实现: 1 ...

  8. java treeset原理_Java集合 --- TreeSet底层实现和原理(源码解析)

    概述 文章的内容基于JDK1.7进行分析,之所以选用这个版本,是因为1.8的有些类做了改动,增加了阅读的难度,虽然是1.7,但是对于1.8做了重大改动的内容,文章也会进行说明. TreeSet实现了S ...

  9. 【dubbo源码解析】--- dubbo中Invoker嵌套调用底层原理

    本文对应源码地址:https://github.com/nieandsun/dubbo-study 文章目录 1 dubbo中Invoker的重要性 2 dubbo RPC链条中代理对象的底层逻辑 2 ...

  10. Colly源码解析——结合例子分析底层实现

    通过<Colly源码解析--框架>分析,我们可以知道Colly执行的主要流程.本文将结合http://go-colly.org上的例子分析一些高级设置的底层实现.(转载请指明出于break ...

最新文章

  1. LeetCode简单题之学生出勤记录 I
  2. [转]控制 Cookie 的作用范围
  3. 解决eclipse中egit中的cannot open git-upload-pack问题
  4. 小知识点——DataTable把满足条件的一行放在第一行
  5. mysql数据库未启动失败_mysql数据库启动失败
  6. Anaconda建立新的环境,出现CondaHTTPError: HTTP 000 CONNECTION FAILED for url 解决过程
  7. 糟糕!服务器被植入挖矿木马,CPU 飙升200%。。。
  8. POJ 2182 Lost Cows [树状数组+二分]
  9. java枚举比较大小写_Spring 3.0 MVC绑定枚举区分大小写
  10. Mybatis中查询结果resultMap使用分析
  11. 黑马程序员_Java集合框架
  12. 投色子--html demo
  13. COM中关于使用DLL的一些知识点
  14. 为什么抢不到红包的总是你?可能是家里路由器没放对
  15. [原]VC被控制时关闭极域电子教室、破解联想硬盘保护系统密码(下)
  16. STM32串口波特率计算问题和常用波特率
  17. 云原生的年代service mesh不止Istio,还有另一个选择-Kuma
  18. 微信支付接口调用之二维码失效时间的设置
  19. 关于汉字与Ascii码
  20. win7 64bit共享XP M1005打印机

热门文章

  1. Atitit table的类型 表类型目录1. 元数据表 日志表 12. 按照文件结构分为堆组织表(HOT)和索引组织表(IOT) 13. 内存表和文件存储表q 24. 全局表 2
  2. Atiti 高并发程序设计 艾提拉著 目录 1. 第—部分 基础知识 2 1.1.  第1章 并行计算机的硬件基础   1.1 并行计算机的组成   1.2 共享存储器多处理器系统   1.3 消
  3. 目录 1.1. Vue是什么??客户端mvc框架,,功能类似springmvc 1 1.2. Why?为什么使用它?? 1 1.3. 包括哪些组件与内部构成与原理 1 2. Howto 怎么使用 2
  4. paip.提升用户体验------c++ 拖曳使用总结..
  5. paip.vs2010 开发ASP浏览时的设置
  6. 石川:未知风险,错误定价,还是数据迁就?
  7. 经纬徐传陞:越是市场低迷之时,越要把握自己的节奏 | 远见2018
  8. 网站建设中常见的21个漏洞及预防方法
  9. 常用数据库的字段类型及大小
  10. VS2015 C#6.0 中的那些新特性(转)