HashMap/ConcurrentHashMap/头插法/尾插法

553 阅读1分钟

1.1 HashMap

JDK1.7

JDK1.8

存储

数组+链表

数组+链表+红黑树

位置算法

h & (length-1)

h & (length-1)

链表超过8

链表

红黑对(链表超过8且数组长度超64)

节点结构

Entry<K,V> implements Map.Entry<K,V>

Node<K,V> implements Map.Entry<K,V>

插法

头插法(扩容环化造成死循环)

尾插法

JDK1.7

使用一个Entry数组来存储数据,用key的hashcode取模来决定key会被放到数组里的位置,如果hashcode取模后的结果相同,那么这些key会被定位到Entry数组的同一个格子里,这些key会形成一个链表;这样数据遍历时间就过长。1.7中hashmap链表插入的方式是使用头插法

JDK1.8

使用一个Node数组来存储数据,但是这个Node可能是链表结构,也可能是红黑树结构;如果插入的元素key的hashcode值相同,那么这些key也会被定位到Node数组的同一个格子里,如果不超过8个使用链表存储,超过8个且Node数组长度超过64,会将链表转换为红黑树。1.8中hashmap链表插入的方式是使用尾插法

【相关问题】

问题一:为什么jdk1.8后改为尾插法?

主要是因为头插法在多线程扩容情况下会引起链表环。那什么是链表环呢?

引用一张图片(来源blog.csdn.net/weixin_3142…):

1.2 ConcurrentHashMap

主要是支持安全的多线程的读写

JDK1.7

JDK1.8

实现

segment+hashentry

Node+CAS+Synchronized

Segment继承ReetrantLock

Synchronized

存储

数组+链表

数组+链表+红黑树

链表超过8

数组

红黑树

插法

头插法

尾插法