|
16 | 16 | > 感谢 [changfubai](https://github.com/changfubai) 对本文的改进做出的贡献!
|
17 | 17 |
|
18 | 18 | ## HashMap 简介
|
19 |
| -HashMap 主要用来存放键值对,它基于哈希表的Map接口实现</font>,是常用的Java集合之一。 |
| 19 | +HashMap 主要用来存放键值对,它基于哈希表的Map接口实现,是常用的Java集合之一。 |
20 | 20 |
|
21 |
| -JDK1.8 之前 HashMap 由 数组+链表 组成的,数组是 HashMap 的主体,链表则是主要为了解决哈希冲突而存在的(“拉链法”解决冲突).JDK1.8 以后在解决哈希冲突时有了较大的变化,当链表长度大于阈值(默认为 8)时,将链表转化为红黑树(将链表转换成红黑树前会判断,如果当前数组的长度小于 64,那么会选择先进行数组扩容,而不是转换为红黑树),以减少搜索时间,具体可以参考 `treeifyBin`方法。 |
| 21 | +JDK1.8 之前 HashMap 由 数组+链表 组成的,数组是 HashMap 的主体,链表则是主要为了解决哈希冲突而存在的(“拉链法”解决冲突)。 |
| 22 | + |
| 23 | +JDK1.8 之后 HashMap 的组成多了红黑树,在满足下面两个条件之后,会执行链表转红黑树操作,以此来加快搜索速度。 |
| 24 | + |
| 25 | +- 链表长度大于阈值(默认为 8) |
| 26 | +- HashMap数组长度超过64å |
22 | 27 |
|
23 | 28 | ## 底层数据结构分析
|
24 | 29 | ### JDK1.8之前
|
25 |
| -JDK1.8 之前 HashMap 底层是 **数组和链表** 结合在一起使用也就是 **链表散列**。**HashMap 通过 key 的 hashCode 经过扰动函数处理过后得到 hash 值,然后通过 `(n - 1) & hash` 判断当前元素存放的位置(这里的 n 指的是数组的长度),如果当前位置存在元素的话,就判断该元素与要存入的元素的 hash 值以及 key 是否相同,如果相同的话,直接覆盖,不相同就通过拉链法解决冲突。** |
| 30 | +JDK1.8 之前 HashMap 底层是 **数组和链表** 结合在一起使用也就是 **链表散列**。 |
| 31 | + |
| 32 | +HashMap 通过 key 的 hashCode 经过扰动函数处理过后得到 hash 值,然后通过 `(n - 1) & hash` 判断当前元素存放的位置(这里的 n 指的是数组的长度),如果当前位置存在元素的话,就判断该元素与要存入的元素的 hash 值以及 key 是否相同,如果相同的话,直接覆盖,不相同就通过拉链法解决冲突。 |
26 | 33 |
|
27 |
| -**所谓扰动函数指的就是 HashMap 的 hash 方法。使用 hash 方法也就是扰动函数是为了防止一些实现比较差的 hashCode() 方法 换句话说使用扰动函数之后可以减少碰撞。** |
| 34 | +所谓扰动函数指的就是 HashMap 的 hash 方法。使用 hash 方法也就是扰动函数是为了防止一些实现比较差的 hashCode() 方法 换句话说使用扰动函数之后可以减少碰撞。 |
28 | 35 |
|
29 | 36 | **JDK 1.8 HashMap 的 hash 方法源码:**
|
30 | 37 |
|
@@ -59,9 +66,11 @@ static int hash(int h) {
|
59 | 66 | 
|
60 | 67 |
|
61 | 68 | ### JDK1.8之后
|
62 |
| -相比于之前的版本,jdk1.8在解决哈希冲突时有了较大的变化,当链表长度大于阈值(默认为8)时,将链表转化为红黑树,以减少搜索时间。 |
| 69 | +相比于之前的版本,JDK1.8 以后在解决哈希冲突时有了较大的变化。 |
63 | 70 |
|
64 |
| - |
| 71 | +当链表长度大于阈值(默认为 8)时,会首先调用 `treeifyBin()`方法,这个方法会根据 HashMap 数组来决定是否转换为红黑树。只有当数组长度大于或者等于 64 的情况下,才会执行转换红黑树操作,以减少搜索时间。否则,就是只是执行 `resize()` 方法对数组扩容。相关源码这里就不贴了,重点关注 `treeifyBin()`方法即可! |
| 72 | + |
| 73 | + |
65 | 74 |
|
66 | 75 | **类的属性:**
|
67 | 76 | ```java
|
@@ -349,8 +358,6 @@ public V put(K key, V value)
|
349 | 358 | }
|
350 | 359 | ```
|
351 | 360 |
|
352 |
| - |
353 |
| - |
354 | 361 | ### get方法
|
355 | 362 | ```java
|
356 | 363 | public V get(Object key) {
|
|
0 commit comments