上篇分析了HashMap的设计思想以及Java7和Java8源码上的实现,固然还有一些"坑"还没填完,好比你们都知道HashMap是线程不安全的数据结构,多线程状况下HashMap会引发死循环引用,它是怎么产生的?Java8引入了红黑树,那是怎么提升效率的?本篇先填第一个坑,仍是以图解的形式加深理解。html
经过上一篇的总体学习,能够知道当存入的键值对超过HashMap的阀值时,HashMap会扩容,即建立一个新的数组,并将原数组里的键值对"转移"到新的数组中。在“转移”的时候,会根据新的数组长度和要转移的键值对key值从新计算在新数组中的位置。重温下Java7中负责"转移"功能的代码java
void transfer(Entry[] newTable, boolean rehash) {
//获取新数组的长度
int newCapacity = newTable.length;
//遍历旧数组中的键值对
for (Entry<K,V> e : table) {
while(null != e) {
Entry<K,V> next = e.next;
if (rehash) {
e.hash = null == e.key ? 0 : hash(e.key);
}
//计算在新表中的索引,并到新数组中
int i = indexFor(e.hash, newCapacity);
e.next = newTable[i];
newTable[i] = e;
e = next;
}
}
}
复制代码
为了加深理解,画个图以下数组
这里假设扩容先后5号坑石头、盖伦、蒙多的hash值与新旧数组长度取模运算后仍是5。上篇文章也总结了,Java7扩容转移先后链表顺序会倒置。当只有单线程操做hashMap时,一切都是那么美好,可是若是多线程同时操做一个hashMap,问题就来了,下面看下多线程操做一个hashMap在Java7源码下是怎样引发死循环引用。安全
前戏是这样的:有两个线程分别叫Thread1和Thread2,它们都有操做同一个hashMap的权利,假设hashMap中的键值对是12个,石头和盖伦扩容先后的hash值与新旧数组长度取模运算后仍是5。扩容前的模拟堆内存状况如图bash
Thread1获得执行权(Thread2被挂起),Thread1往hashMap里put第13个键值对的时候判断超过阀值,执行扩容操做,Thread1建立了一个新数组,还没来得及转移旧键值对的时候,系统时间片反手切到Thread2(Thread1被挂起),整个过程用图表示数据结构
能够看到Thread1只是建立了个新数组,还没来得及转移就被挂起了,新数组没有内容,此时在Thread1的视角认的是e是石头,next是盖伦;此时的模拟内存图状况多线程
再看下Thread2的操做,一样Thread2往hashMap里put第13个键值对的时候判断超过阀值,执行扩容操做,Thread2先建立一个新数组,不一样的是,Thread2运气好,在时间片轮换前转移工做也走完了。第一次遍历oop
第二次遍历源码分析
此时模拟的内存状况学习
能够看到此时对于盖伦来讲,他的next是石头;对于石头来讲,它的next为null,隐患就此埋下。接下来时间片又切到Thread1(停了半天终于轮到我出场了),先看下Thread1的处境
结合代码分析以下
第一步:
第二步:
第三步:
第四步:
第五步:
第六步:
第七步:
第八步:
第九步:
第10步:
到这终于看到盖伦和石头"互指",水乳交融。
那这会带来什么后果呢?后续操做新数组的5号坑会进入死循环(注意,操做其余坑并不会有问题),例如Java7 put操做
Java7 get操做会执行getEntry,一样会引发死循环。
到此,Java7多线程操做HashMap可能造成死循环的缘由剖析完成。
经过上一篇的学习可知,Java7转移先后位置颠倒,而Java8转移键值对先后位置不变。一样的前戏,看下代码
此时模拟堆内存状况
Thread1的状况
这时候Thread2得到执行权,扩容并完成转移工做,经过上篇的学习可知,Java8在转移前会建立两条链表,即扩容后位置不加原数组长度的lo链和要加原数组长度的hi链,这里假设石头和盖伦扩容先后都在5号坑,即这是一条lo链(其实就算不在同一个坑也不影响,缘由就是Java8扩容先后链顺序不变)。Thread2遍历第一次
第二次
能够看到Thread2全程是没有去修改石头和盖伦的引用关系,石头.next是盖伦,盖伦.next是null。那么Thread1获得执行权后其实只是重复了Thread2的工做。
经过源码分析,Java7在多线程操做hashmap时可能引发死循环,缘由是扩容转移后先后链表顺序倒置,在转移过程当中修改了原来链表中节点的引用关系;Java8在一样的前提下并不会引发死循环,缘由是扩容转移后先后链表顺序不变,保持以前节点的引用关系。那是否是意味着Java8就能够把HashMap用在多线程中呢?我的感受即便不会出现死循环,可是经过源码看到put/get方法都没有加同步锁,多线程状况最容易出现的就是:没法保证上一秒put的值,下一秒get的时候仍是原值,建议使用ConcurrentHashMap。