Skip to content

Commit 11ebd35

Browse files
committed
Update HashMap(JDK1.8)源码+底层数据结构分析.md
1 parent e029a36 commit 11ebd35

File tree

1 file changed

+15
-8
lines changed

1 file changed

+15
-8
lines changed

docs/java/collection/HashMap(JDK1.8)源码+底层数据结构分析.md

Lines changed: 15 additions & 8 deletions
Original file line numberDiff line numberDiff line change
@@ -16,15 +16,22 @@
1616
> 感谢 [changfubai](https://github.com/changfubai) 对本文的改进做出的贡献!
1717
1818
## HashMap 简介
19-
HashMap 主要用来存放键值对,它基于哈希表的Map接口实现</font>,是常用的Java集合之一。
19+
HashMap 主要用来存放键值对,它基于哈希表的Map接口实现,是常用的Java集合之一。
2020

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å
2227

2328
## 底层数据结构分析
2429
### 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 是否相同,如果相同的话,直接覆盖,不相同就通过拉链法解决冲突。
2633

27-
**所谓扰动函数指的就是 HashMap 的 hash 方法。使用 hash 方法也就是扰动函数是为了防止一些实现比较差的 hashCode() 方法 换句话说使用扰动函数之后可以减少碰撞。**
34+
所谓扰动函数指的就是 HashMap 的 hash 方法。使用 hash 方法也就是扰动函数是为了防止一些实现比较差的 hashCode() 方法 换句话说使用扰动函数之后可以减少碰撞。
2835

2936
**JDK 1.8 HashMap 的 hash 方法源码:**
3037

@@ -59,9 +66,11 @@ static int hash(int h) {
5966
![jdk1.8之前的内部结构](https://my-blog-to-use.oss-cn-beijing.aliyuncs.com/2019-7/jdk1.8之前的内部结构.png)
6067

6168
### JDK1.8之后
62-
相比于之前的版本,jdk1.8在解决哈希冲突时有了较大的变化,当链表长度大于阈值(默认为8)时,将链表转化为红黑树,以减少搜索时间
69+
相比于之前的版本,JDK1.8 以后在解决哈希冲突时有了较大的变化
6370

64-
![JDK1.8之后的HashMap底层数据结构](http://my-blog-to-use.oss-cn-beijing.aliyuncs.com/18-8-22/67233764.jpg)
71+
当链表长度大于阈值(默认为 8)时,会首先调用 `treeifyBin()`方法,这个方法会根据 HashMap 数组来决定是否转换为红黑树。只有当数组长度大于或者等于 64 的情况下,才会执行转换红黑树操作,以减少搜索时间。否则,就是只是执行 `resize()` 方法对数组扩容。相关源码这里就不贴了,重点关注 `treeifyBin()`方法即可!
72+
73+
![](https://oscimg.oschina.net/oscnet/up-bba283228693dae74e78da1ef7a9a04c684.png)
6574

6675
**类的属性:**
6776
```java
@@ -349,8 +358,6 @@ public V put(K key, V value)
349358
}
350359
```
351360

352-
353-
354361
### get方法
355362
```java
356363
public V get(Object key) {

0 commit comments

Comments
 (0)