描述

  • 可以按照添加元素的顺序对元素进行迭代的HashMap的子类.
  • 注意,上面说的是加元素的顺序.也就是说,更新元素时,是不会影响遍历结构的的.除非设置参数accessOrdertrue,将更新元素放置到队末.
  • 这个类没有对其父类HashMap进行过多重写.主要通过实现afterNode*相关方法,在数据结构变更后,进行后置的链表结构更新进行维护.

常用与关键方法

linkNodeLast方法

描述:

  • 负责初始化成员变量headtail.
  • headtail初始化完成后,负责将目标元素p连接到tail并更新原有tail到目标元素p

代码:

private void linkNodeLast(LinkedHashMap.Entry<K,V> p) {
// 缓存尾部
LinkedHashMap.Entry<K,V> last = tail;
// 更新尾部到新元素
tail = p;
// 判断老尾部是否已经初始化
if (last == null)
// 老尾部为初始化,代表头部也没初始化.进行初始化操作
head = p;
else {
// 初始化以完成,将p链接到老尾部之后
p.before = last;
last.after = p;
}
}

transferLinks方法

描述:

使用dst替换src在双向链表中的位置

代码:

private void transferLinks(LinkedHashMap.Entry<K,V> src,
LinkedHashMap.Entry<K,V> dst) {
// 同步before,同时保存到局部变量
LinkedHashMap.Entry<K,V> b = dst.before = src.before;
// 同步after,同时保存到局部变量
LinkedHashMap.Entry<K,V> a = dst.after = src.after;
// 检查before
if (b == null)
// 没有before.将dst设置为head节点
head = dst;
else
// 有before,将before与dst关联
b.after = dst;
// 检查after
if (a == null)
// 没有after,将dst作为tail节点
tail = dst;
else
// 有after,将after与dst连接
a.before = dst;
}

newNode方法

描述:

重写了父类newNode方法.扩展双向链表的连接操作.返回了HashMap.Node的子类节点LinkedHashMap.Entry.

代码:

Node<K,V> newNode(int hash, K key, V value, Node<K,V> e) {
LinkedHashMap.Entry<K,V> p =
new LinkedHashMap.Entry<K,V>(hash, key, value, e);
// 创建的新节点.直接链接到末端节点上
linkNodeLast(p);
return p;
}

replacementNode方法

描述:

扩展双向链表替换节点的操作.这个方法用于父类HashMapHashMap.TreeNode替换为HashMap.Node时调用,这里进行了重写,使用带有双向链表的LinkedHashMap.Entry作为返回值

注意: 这里HashMap.TreeNode是实现了LinkedHashMap.Entry的.也就是参数p,他可以直接强转为实现类LinkedHashMap.Entry

代码:

Node<K,V> replacementNode(Node<K,V> p, Node<K,V> next) {
LinkedHashMap.Entry<K,V> q = (LinkedHashMap.Entry<K,V>)p;
LinkedHashMap.Entry<K,V> t =
new LinkedHashMap.Entry<K,V>(q.hash, q.key, q.value, next);
// 替换节点
transferLinks(q, t);
return t;
}

newTreeNode方法

描述:

重写了父类方法newTreeNode.扩展双向链表的连接操作.同样,因为HashMap.TreeNode实现LinkedHashMap.Entry.可以直接通过linkNodeLast方法进行连接操作

代码:

TreeNode<K,V> newTreeNode(int hash, K key, V value, Node<K,V> next) {
TreeNode<K,V> p = new TreeNode<K,V>(hash, key, value, next);
linkNodeLast(p);
return p;
}

replacementTreeNode方法

描述:

replacementNode.扩展双向链表替换节点的操作.只是节点类型变成了TreeNode.又因为他是LinkedHashMap.Entry的子类,可以直接交给transferLinks使用.进行双向链表替换操作

代码:

TreeNode<K,V> replacementTreeNode(Node<K,V> p, Node<K,V> next) {
LinkedHashMap.Entry<K,V> q = (LinkedHashMap.Entry<K,V>)p;
TreeNode<K,V> t = new TreeNode<K,V>(q.hash, q.key, q.value, next);
transferLinks(q, t);
return t;
}

afterNodeRemoval方法

描述:

删除节点后调用.进行双向链表同步

代码:

void afterNodeRemoval(Node<K,V> e) { // unlink
// b - before节点
// p - 被删除节点
// a - after节点
LinkedHashMap.Entry<K,V> p =
(LinkedHashMap.Entry<K,V>)e, b = p.before, a = p.after;
// 清除p的双端引用
p.before = p.after = null; // 判断before是否存在
if (b == null)
// 没有before
// 设置a为head
head = a;
else
// 存在before
// 连接b->a.注意,这是单向连接,现在还无法确认a是否存在.如果a为空,b就是链表中的唯一节点.after属性为null
b.after = a;
// 判断a是否为空
if (a == null)
// a为空
// tail设置为b
tail = b;
else
// a存在
// 连接 a->b.注意,这里也是单向连接.如果b是空的话,a现在就是head且before属性是null
a.before = b;
}

afterNodeAccess方法

描述:

更新节点后调用.进行双向链表同步

代码:

void afterNodeAccess(Node<K,V> e) { // move node to last
// oldTail.老尾部缓存
LinkedHashMap.Entry<K,V> last;
// 判断accessOrder.即按照访问(更新)顺序排列
// 获取老尾部
// 判断当前元素是不是尾部元素
if (accessOrder && (last = tail) != e) {
// accessOrder==true且e不要尾部元素 // b - fefore
// p - 当前元素
// a - after
LinkedHashMap.Entry<K,V> p =
(LinkedHashMap.Entry<K,V>)e, b = p.before, a = p.after; // 因为p将变为尾部元素,所以直接设置p.after为null.
p.after = null; // 判断b
if (b == null)
// b为null,p节点就是head节点
// a作为头部节点
head = a;
else
// b不为空
// 连接b->a. 注意,这里是单向连接.a可能为null,a.before的连接交给后续判断
b.after = a; // 判断a
if (a != null)
// a不为空
// a->b.注意,这里是单向链接.b可能是null.b.after的连接交给后续判断
a.before = b;
else
// a为空.p节点就是tail节点
// 这里有两个分支,需要判断b是否为空.此处a已经为空,如果b也为空,说明p是列表中的唯一节点.这个判断委托到后续判断中处理
// 此时,last变量已经失去意义,它与p为同一对象.
// 这里说一下赋值last = b;的作用.注意,这是本人猜测!
// 是为了统一算法的外在样式.因为变量last在在本方法中是不会为空的,且在所有的情形中,都会调用p.before = last;last.after = p;进行连接(除了p是唯一元素的情况).
// 那么在b存在的时候,再次与p进行连接,在链表结构上也是没有问题的,统一被视作被操作元素的前一个元素
last = b;
if (last == null)
// p是唯一元素
head = p;
else {
// 连接到尾部节点
p.before = last;
last.after = p;
}
// 更新尾部节点到p
tail = p;
// 修改计数++
++modCount;
}
}

内部类

LinkedHashIterator

描述:

封装了针对链表结构的迭代器.并向子类提供了共有的扩展方法.

代码:

abstract class LinkedHashIterator {
LinkedHashMap.Entry<K,V> next;
LinkedHashMap.Entry<K,V> current;
int expectedModCount; LinkedHashIterator() {
// 初始化next节点为当前head
next = head;
expectedModCount = modCount;
current = null;
} public final boolean hasNext() {
return next != null;
} final LinkedHashMap.Entry<K,V> nextNode() {
// 缓存next
LinkedHashMap.Entry<K,V> e = next;
// fast-fail
if (modCount != expectedModCount)
throw new ConcurrentModificationException();
// next为空
if (e == null)
throw new NoSuchElementException();
// 设置当前
current = e;
// 更新next到下一个
next = e.after;
return e;
} public final void remove() {
// 获取当前
Node<K,V> p = current;
// null判断
if (p == null)
throw new IllegalStateException();
// fast-fail
if (modCount != expectedModCount)
throw new ConcurrentModificationException();
// 迭代器置空
current = null;
// 获取key
K key = p.key;
// 调用父类的removeNode方法进行节点删除
removeNode(hash(key), key, null, false, false);
// 同步更新计数
expectedModCount = modCount;
}
}

内部类

LinkedHashIterator实现

描述:

分别继承了LinkedHashIterator并使用前者的nextNode方法返回不同数据

代码:

final class LinkedKeyIterator extends LinkedHashIterator
implements Iterator<K> {
public final K next() { return nextNode().getKey(); }
} final class LinkedValueIterator extends LinkedHashIterator
implements Iterator<V> {
public final V next() { return nextNode().value; }
} final class LinkedEntryIterator extends LinkedHashIterator
implements Iterator<Map.Entry<K,V>> {
public final Map.Entry<K,V> next() { return nextNode(); }
}

LinkedHashMap源码学习的更多相关文章

  1. JDK源码学习笔记——LinkedHashMap

    HashMap有一个问题,就是迭代HashMap的顺序并不是HashMap放置的顺序,也就是无序. LinkedHashMap保证了元素迭代的顺序.该迭代顺序可以是插入顺序或者是访问顺序.通过维护一个 ...

  2. Java集合专题总结(1):HashMap 和 HashTable 源码学习和面试总结

    2017年的秋招彻底结束了,感觉Java上面的最常见的集合相关的问题就是hash--系列和一些常用并发集合和队列,堆等结合算法一起考察,不完全统计,本人经历:先后百度.唯品会.58同城.新浪微博.趣分 ...

  3. 转:【Java集合源码剖析】LinkedHashmap源码剖析

    转载请注明出处:http://blog.csdn.net/ns_code/article/details/37867985   前言:有网友建议分析下LinkedHashMap的源码,于是花了一晚上时 ...

  4. hashMap源码学习记录

    hashMap作为java开发面试最常考的一个题目之一,有必要花时间去阅读源码,了解底层实现原理. 首先,让我们看看hashMap这个类有哪些属性 // hashMap初始数组容量 static fi ...

  5. linkedHashMap源码解析(JDK1.8)

    引言 关于java中的不常见模块,让我一下子想我也想不出来,所以我希望以后每次遇到的时候我就加一篇.上次有人建议我写全所有常用的Map,所以我研究了一晚上LinkedHashMap,把自己感悟到的解释 ...

  6. HashMap的源码学习以及性能分析

    HashMap的源码学习以及性能分析 一).Map接口的实现类 HashTable.HashMap.LinkedHashMap.TreeMap 二).HashMap和HashTable的区别 1).H ...

  7. JDK1.8源码学习-HashMap

    JDK1.8源码学习-HashMap 目录 一.HashMap简介 HashMap 主要用来存放键值对,它是基于哈希表的Map接口实现的,是常用的Java集合之一. 我们都知道在JDK1.8 之前 的 ...

  8. jQuery源码学习感想

    还记得去年(2015)九月份的时候,作为一个大四的学生去参加美团霸面,结果被美团技术总监教育了一番,那次问了我很多jQuery源码的知识点,以前虽然喜欢研究框架,但水平还不足够来研究jQuery源码, ...

  9. MVC系列——MVC源码学习:打造自己的MVC框架(四:了解神奇的视图引擎)

    前言:通过之前的三篇介绍,我们基本上完成了从请求发出到路由匹配.再到控制器的激活,再到Action的执行这些个过程.今天还是趁热打铁,将我们的View也来完善下,也让整个系列相对完整,博主不希望烂尾. ...

随机推荐

  1. 大型情感剧集Selenium:2_options设置 #华为云·寻找黑马程序员#

    上集回顾 昨天说简单介绍了什么是selenium,它能干what,和发展史与梗概.当的是python如何通过pip安装selenium,并下载对应浏览器的webdriver. 最后简单通过一个Demo ...

  2. ASP.NET Core 选项模式源码学习Options IOptions(二)

    前言 上一篇文章介绍IOptions的注册,本章我们继续往下看 IOptions IOptions是一个接口里面只有一个Values属性,该接口通过OptionsManager实现 public in ...

  3. JDBC技术对数据库进行操作

    什么是 JDBC: • JDBC(Java DataBase Connectivity)java 数据库连接 • 是 JavaEE 平台下的技术规范 • 定义了在 Java 语言中连接数据,执行 SQ ...

  4. vuex简单使用。

    项目结构: 1:首先在项目中新建store.js文件,.js文件内容如下: import Vue from 'vue' import Vuex from 'vuex' Vue.use(Vuex) ex ...

  5. ELK filebeat的安装

    安装参考官方文档: https://www.elastic.co/guide/en/beats/filebeat/current/filebeat-installation.html 注意事项: Fr ...

  6. 洛谷 P2764(最小路径覆盖=节点数-最大匹配)

    给定有向图G=(V,E).设P 是G 的一个简单路(顶点不相交)的集合.如果V 中每个顶点恰好在P 的一条路上,则称P是G 的一个路径覆盖.P 中路径可以从V 的任何一个顶点开始,长度也是任意的,特别 ...

  7. seaborn 数据可视化(二)带有类别属性的数据可视化

    Seaborn的分类图分为三类,将分类变量每个级别的每个观察结果显示出来,显示每个观察分布的抽象表示,以及应用统计估计显示的权重趋势和置信区间: 第一个包括函数swarmplot()和stripplo ...

  8. BGA256芯片植球全过程体验(原创)

    今天工具到位,迫不亟待,需要对手上的BGA256的FPGA芯片进行植球, 该芯片买来的时候是有球的,只是在焊接后,由于电路板故障或焊接问题,需要拆下来芯片,导致球损失,需要重新植球. 一般植球都是将所 ...

  9. 10分钟理解BFC原理

    10 分钟理解 BFC 原理 一.常见定位方案 在讲 BFC 之前,我们先来了解一下常见的定位方案,定位方案是控制元素的布局,有三种常见方案: 普通流 (normal flow) 在普通流中,元素按照 ...

  10. UWP 使用SSL证书,保证数据安全

    事情是这样的,我们后端的小伙伴升级了用户会员系统,使用了全新的GraphQL登录机制,并且采用SSL加密的方式来实现阻止陌生客户端请求的案例. GraphQL在UWP端的实现,以后有时间会单独写一篇文 ...