目录

​HashMap和HashTable区别(HashTable线程安全)​

​HashTable​

​HashMap​

​ConcurrentHashMap (已经放弃)​

 

​HashMap原理​

​1、put方法原理​

​ 2、get方法原理​

​HashMap 扩容(resize)​

​HashMap的高并发问题(线程不安全)​

​哈希表原理:​



 

HashMap和HashTable区别(HashTable线程安全)

 

HashTable

底层数组+链表实现,无论key还是value都不能为null,线程安全,实现线程安全的方式是在修改数据时锁住整个HashTable,效率低,ConcurrentHashMap做了相关优化

初始size为11,扩容:newsize = oldsize*2+1

计算index的方法:index = (hash & 0x7FFFFFFF) % tab.length

HashMap

底层数组+链表实现,可以存储null键和null值,线程不安全

初始size为16,扩容:newsize = oldsize*2,size一定为2的n次幂

扩容针对整个Map,每次扩容时,原来数组中的元素依次重新计算存放位置,并重新插入

插入元素后才判断该不该扩容,有可能无效扩容(插入后如果扩容,如果没有再次插入,就会产生无效扩容)

当Map中元素总数超过Entry数组的75%,触发扩容操作,为了减少链表长度,元素分配更均匀

计算index方法:index = hash & (tab.length – 1)

HashMap的初始值还要考虑加载因子:

哈希冲突:若干Key的哈希值按数组大小取模后,如果落在同一个数组下标上,将组成一条Entry链,对Key的查找需要遍历Entry链上的每个元素执行equals()比较。

加载因子:为了降低哈希冲突的概率,默认当HashMap中的键值对达到数组大小的75%时,即会触发扩容。因此,如果预估容量是100,即需要设定100/0.75=134的数组大小。

空间换时间:如果希望加快Key查找的时间,还可以进一步降低加载因子,加大初始大小,以降低哈希冲突的概率。

ConcurrentHashMap (已经放弃)

底层采用分段的数组+链表实现,线程安全

通过把整个Map分为N个Segment,可以提供相同的线程安全,但是效率提升N倍,默认提升16倍。(读操作不加锁,由于HashEntry的value变量是 volatile的,也能保证读取到最新的值。)

Hashtable的synchronized是针对整张Hash表的,即每次锁住整张表让线程独占,ConcurrentHashMap允许多个修改操作并发进行,其关键在于使用了锁分离技术

有些方法需要跨段,比如size()和containsValue(),它们可能需要锁定整个表而而不仅仅是某个段,这需要按顺序锁定所有段,操作完毕后,又按顺序释放所有段的锁

扩容:段内扩容(段内元素超过该段对应Entry数组长度的75%触发扩容,不会对整个Map进行扩容),插入前检测需不需要扩容,有效避免无效扩容

 HashMap原理

(更通俗的解释:​​https://zhuanlan.zhihu.com/p/79507868​​)

HashMap的数据结构

HashMap的数据结构为 数组+(链表或红黑树),上图

【hashmap】HashMap原理及线程不安全详解|哈希表原理_键值对

HashMap是一个用于存储Key-Value键值对的集合,每一个键值对也叫做Entry。这些个键值对(Entry)分散存储在一个数组当中,数组初始化长度为16,数组中的每一个元素初始值都是null,这个数组就是HashMap的主干。

1、put方法原理(存储过程)

当调用hashMap.put("apple", 0) ,插入一个Key为“apple"的元素。这时候我们需要利用一个哈希函数来确定Entry的插入位置(index):

index =  Hash(“apple”)-----Hash中哈希函数计算出hashcode,再用取模(hashcode/数组长度)等方法得到index

假定最后计算出的index是2,那么结果如下:

【hashmap】HashMap原理及线程不安全详解|哈希表原理_数组长度_02

但是,因为HashMap的长度是有限的,当插入的Entry越来越多时,再完美的Hash函数也难免会出现index冲突的情况。比如下面这样:

【hashmap】HashMap原理及线程不安全详解|哈希表原理_键值对_03

HashMap数组的每一个元素不止是一个Entry对象,也是一个链表的头节点。每一个Entry对象通过Next指针指向它的下一个Entry节点。当新来的Entry映射到冲突的数组位置时,只需要插入到对应的链表即可:

【hashmap】HashMap原理及线程不安全详解|哈希表原理_链表_04

需要注意的是,新来的Entry节点插入链表时,使用的是“头插法”。 原因时HashMap的设计者认为后插入的元素被get的几率更大。

 2、get方法原理

使用Get方法根据Key来查找Value的时候,发生了什么呢?

首先会把输入的Key做一次Hash映射,得到对应的index:

index =  Hash(“apple”)

由于刚才所说的Hash冲突,同一个位置有可能匹配到多个Entry,这时候就需要顺着对应链表的头节点,一个一个向下来查找。假设我们要查找的Key是“apple”:

【hashmap】HashMap原理及线程不安全详解|哈希表原理_键值对_05

第一步,我们查看的是头节点Entry6,Entry6的Key是banana,显然不是我们要找的结果。

第二步,我们查看的是Next节点Entry1,Entry1的Key是apple,正是我们要找的结果。

之所以把Entry6放在头节点,是因为HashMap的发明者认为,后插入的Entry被查找的可能性更大。

 

HashMap 扩容(resize)

HashMap的默认初始长度时16,并且每次自动扩展或者手动初始化时,长度必须时2的幂

原因:

在于HashMap的数组下标算法,当put一个值时,会首先计算出hash code,根据hash code决定放在数组的位置,考虑效率问题并没有采用取模的方式,而是采用按位与的方式

下面我们以“book"的Key来演示整个过程:

1.计算book的hashcode,结果为十进制的3029737,二进制的101110001110101110 1001。

2.假定HashMap长度是默认的16,计算Length-1的结果为十进制的15,二进制的1111。

3.把以上两个结果做与运算,101110001110101110 1001 & 1111 = 1001,十进制是9,所以 index=9。

可以说,Hash算法最终得到的index结果,完全取决于Key的Hashcode值的最后几位。

在这种情况下,如果数字的长度不是2的幂,会发生以下结果:

假设HashMap的长度是10,重复刚才的运算步骤:

【hashmap】HashMap原理及线程不安全详解|哈希表原理_数组_06

单独看这个结果,表面上并没有问题。我们再来尝试一个新的HashCode  101110001110101110 1011 :

【hashmap】HashMap原理及线程不安全详解|哈希表原理_键值对_07

让我们再换一个HashCode 101110001110101110 1111 试试  :

【hashmap】HashMap原理及线程不安全详解|哈希表原理_数组_08

虽然HashCode的倒数第二第三位从0变成了1,但是运算的结果都是1001。也就是说,当HashMap长度为10的时候,有些index结果的出现几率会更大,而有些index结果永远不会出现(比如0111)! 这样,显然不符合Hash算法均匀分布的原则。 反观长度16或者其他2的幂,Length-1的值是所有二进制位全为1,这种情况下,index的结果等同于HashCode后几位的值。只要输入的HashCode本身分布均匀,Hash算法的结果就是均匀的。

 

HashMap的高并发问题(线程不安全)

HashMap是线程不安全的,原因就在于HashMap的rehash。rehash是HashMap扩容过程种的一个步骤。 HashMap的容量是有限的。

当经过多次元素插入,使得HashMap达到一定饱和度时,Key映射位置发生冲突的几率会逐渐提高。 这时候,HashMap需要扩展它的长度,也就是进行Resize。 影响发生Resize的因素有两个:

1.Capacity

HashMap的当前长度。上一期曾经说过,HashMap的长度是2的幂。

2.LoadFactor

HashMap负载因子,默认值为0.75f。

衡量HashMap是否进行Resize的条件如下:

HashMap.Size   >=  Capacity * LoadFactor

 

HashMap的扩容主要分为两步:

1.扩容

创建一个新的Entry空数组,长度是原数组的2倍

2.ReHash

遍历原Entry数组,把所有的Entry重新Hash到新数组。为什么要重新Hash呢?因为长度扩大以后,Hash的规则也随之改变。

让我们回顾一下Hash公式:

index =  HashCode(Key) &  (Length - 1)

当原数组长度为8时,Hash运算是和111B做与运算;新数组长度为16,Hash运算是和1111B做与运算。Hash结果显然不同。

ReHash的Java代码如下:

/**
* Transfers all entries from current table to newTable.
*/
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;
}
}
}

以上过程单线程下不会出现问题,但是当两个线程同时触发resize的时候就有可能出现问题

假设一个HashMap已经到了Resize的临界点。此时有两个线程A和B,在同一时刻对HashMap进行Put操作:

【hashmap】HashMap原理及线程不安全详解|哈希表原理_链表_09

此时达到Resize条件,两个线程各自进行Rezie的第一步,也就是扩容:

【hashmap】HashMap原理及线程不安全详解|哈希表原理_链表_10

这时候,两个线程都走到了ReHash的步骤。让我们回顾一下ReHash的代码:

【hashmap】HashMap原理及线程不安全详解|哈希表原理_线程安全_11

假如此时线程B遍历到Entry3对象,刚执行完红框里的这行代码,线程就被挂起。对于线程B来说:

e = Entry3

next = Entry2

这时候线程A畅通无阻地进行着Rehash,当ReHash完成后,结果如下(图中的e和next,代表线程B的两个引用):

【hashmap】HashMap原理及线程不安全详解|哈希表原理_数组长度_12

直到这一步,看起来没什么毛病。接下来线程B恢复,继续执行属于它自己的ReHash。线程B刚才的状态是:

e = Entry3

next = Entry2

【hashmap】HashMap原理及线程不安全详解|哈希表原理_线程安全_13

当执行到上面这一行时,显然 i = 3,因为刚才线程A对于Entry3的hash结果也是3。

【hashmap】HashMap原理及线程不安全详解|哈希表原理_数组长度_14

我们继续执行到这两行,Entry3放入了线程B的数组下标为3的位置,并且e指向了Entry2。此时e和next的指向如下:

e = Entry2

next = Entry2

整体情况如图所示:

【hashmap】HashMap原理及线程不安全详解|哈希表原理_链表_15

接着是新一轮循环,又执行到红框内的代码行:

【hashmap】HashMap原理及线程不安全详解|哈希表原理_数组长度_16

e = Entry2

next = Entry3

整体情况如图所示:

【hashmap】HashMap原理及线程不安全详解|哈希表原理_数组长度_17

接下来执行下面的三行,用头插法把Entry2插入到了线程B的数组的头结点:

【hashmap】HashMap原理及线程不安全详解|哈希表原理_数组_18

整体情况如图所示:

【hashmap】HashMap原理及线程不安全详解|哈希表原理_数组长度_19

第三次循环开始,又执行到红框的代码:

【hashmap】HashMap原理及线程不安全详解|哈希表原理_线程安全_20

e = Entry3

next = Entry3.next = null

最后一步,当我们执行下面这一行的时候,见证奇迹的时刻来临了:

【hashmap】HashMap原理及线程不安全详解|哈希表原理_数组长度_21

newTable[i] = Entry2

e = Entry3

Entry2.next = Entry3

Entry3.next = Entry2

链表出现了环形!

整体情况如图所示:

【hashmap】HashMap原理及线程不安全详解|哈希表原理_键值对_22

此时,问题还没有直接产生。当调用Get查找一个不存在的Key,而这个Key的Hash结果恰好等于3的时候,由于位置3带有环形链表,所以程序将会进入死循环!

 ​

解释2:​​https://coolshell.cn/articles/9606.html​

 

哈希表原理: