HashMap源码解析
2021/10/13 14:14:49
本文主要是介绍HashMap源码解析,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!
Java源码解析之HashMap
一、HashMap源码解析
1、HashMap的数据结构
- jdk7以前:数组+链表
- jdk8以后:数组+链表+红黑树
public class HashMap<K,V> extends AbstractMap<K,V> implements Map<K,V>, Cloneable, Serializable { static class Node<K,V> implements Map.Entry<K,V> { final int hash; final K key; V value; Node<K,V> next; ... ... } transient Node<K,V>[] table; }
以上代码是jdk中HashMap中的部分代码
HashMap类中定义了一个名为Node的静态内部类,它实现了Map.Entry接口,这个内部类实例化后就是一个Entry的实现类对象,每个
Entry对象中有指向下一个元素的Node<K,V> next属性,这就是所谓的链表的实现方式(通过地址指引)
HashMap类中还定义了一个Node<K,V>[] table的数组,用于存储一个个的Node对象
从这里可以看出HashMap中存储的起始是一个个的Entry对象
小知识点
序列化:一个类实现了Serializable,就相当于告诉JVM这个类是可以被序列化的,就是可以把对象通过流写出
transient关键字:在一个可序列化的类中,transient关键字修饰的成员表示该成员不参与序列化,也就是不能写出保存,相当 于是一个临时数据
反序列化:就是把序列化写出的对象读到程序内存中
下面是Node内部类的具体源代码
static class Node<K,V> implements Map.Entry<K,V> { final int hash; final K key; V value; Node<K,V> next; Node(int hash, K key, V value, Node<K,V> next) { this.hash = hash; this.key = key; this.value = value; this.next = next; } public final K getKey() { return key; } public final V getValue() { return value; } public final String toString() { return key + "=" + value; } public final int hashCode() { return Objects.hashCode(key) ^ Objects.hashCode(value); } public final V setValue(V newValue) { V oldValue = value; value = newValue; return oldValue; } public final boolean equals(Object o) { if (o == this) return true; if (o instanceof Map.Entry) { Map.Entry<?,?> e = (Map.Entry<?,?>)o; if (Objects.equals(key, e.getKey()) && Objects.equals(value, e.getValue())) return true; } return false; } }
2、HashMap的构造方法
public class HashMap<K,V> extends AbstractMap<K,V> implements Map<K,V>, Cloneable, Serializable { static final int DEFAULT_INITIAL_CAPACITY = 1 << 4; static final int MAXIMUM_CAPACITY = 1 << 30; static final float DEFAULT_LOAD_FACTOR = 0.75f; transient int modCount; //记录这个HashMap结构被修改的次数 transient int size; //这个HashMap中包含键值对的数量 int threshold; //下一次扩容的条件数值,当数组内容达到这个值时进行扩容 }
HashMap类存在这三种常量属性,和哈希表的性能有关,分别是
static final int DEFAULT_INITIAL_CAPACITY = 1 << 4
:定义了Node<K,V>[] table
数组的默认大小
static final int MAXIMUM_CAPACITY = 1 << 30
:定义了Node<K,V>[] table
数组的最大长度
static final float DEFAULT_LOAD_FACTOR = 0.75f
:定义了哈希表的加载因子
1 << 4 << 是移位操作,是基于二进制的操作
1的二进制 0000 0001 就是十进制的1
1 << 4 后 0001 0000 就是十进制的16
加载因子:加载因子就是判断数组什么时候进行扩容,当数组内元素个数满足 原数组长度*加载因子 个时进行扩容
-
空构造public HashMap()
public HashMap() { this.loadFactor = DEFAULT_LOAD_FACTOR; // all other fields defaulted }
调用空构造函数时,生成对象的字段属性使用的都是默认值
数组默认长度16bit、加载因子0.75
-
包含数组长度和加载因子的构造函数public HashMap(int initialCapacity, float loadFactor)
public HashMap(int initialCapacity, float loadFactor) { if (initialCapacity < 0) throw new IllegalArgumentException("Illegal initial capacity: " + initialCapacity); if (initialCapacity > MAXIMUM_CAPACITY) initialCapacity = MAXIMUM_CAPACITY; if (loadFactor <= 0 || Float.isNaN(loadFactor)) throw new IllegalArgumentException("Illegal load factor: " + loadFactor); this.loadFactor = loadFactor; this.threshold = tableSizeFor(initialCapacity); }
这个构造方法中,传了一个数组的长度和加载因子的数组作为参数,函数内就是判断这两个值是否符合条件,符合的话就为对象的loadFactor属性赋值,并使用 this.threshold = tableSizeFor(initialCapacity);为threshold 赋值;否则报错
tableSizeFor这个函数返回一个大于指定值得最小2次幂
使用 this.threshold = tableSizeFor(initialCapacity)为threshold 赋值???
threshold的值不应该是 this.threshold = tableSizeFor(initialCapacity)*loadFactor吗?
在HashMap的构造函数,并没有为Node<K,V>[] table进行初始化,而是把初始化的过程放在了put方法里
-
包含数组长度的构造函数public HashMap(int initialCapacity)
public HashMap(int initialCapacity) { this(initialCapacity, DEFAULT_LOAD_FACTOR); }
这个就是调用public HashMap(int initialCapacity, float loadFactor)构造,把加载因子为默认值
3、HashMap的存值(put方法)
public V put(K key, V value) { return putVal(hash(key), key, value, false, true); }
调用put方法时,put方法会把key的hash值计算出来,然后调用putVal方法,返回值为Value的值
-
1.putVal方法会先判断数组是否需要扩容
final V putVal(int hash, K key, V value, boolean onlyIfAbsent, boolean evict) { Node<K,V>[] tab; Node<K,V> p; int n, i; if ((tab = table) == null || (n = tab.length) == 0) n = (tab = resize()).length;
如果数组为空或者长度为零,则调用resize方法扩容数组,这里才对数组进行初始化操作
-
2.然后计算出元素的索引,判断数组上该位置是否已经存过元素
if ((p = tab[i = (n - 1) & hash]) == null) tab[i] = newNode(hash, key, value, null);
如果该位置为空,则直接存入该位置
(n-1) & hash 是求该hash值在数组中的索引的操作
n ---> 数组长度
例如:数组长度为16 hash值为一个随机值
n 0001 0000 16
n-1 0000 1111 15
hash 1011 0011 随机
& 0000 0011 &后的结果 3
(n-1) & hash相当于拿出一个随机值的后几位
-
3.如果数组当前索引处已经存有元素了,把数组当前索引存的对象称为 p
-
判断这个元素的key值是否和p的key值相同
else { Node<K,V> e; K k; if (p.hash == hash && ((k = p.key) == key || (key != null && key.equals(k)))) e = p;
如果相同则取出p对象,否则
-
判断p是否为一个树节点
else if (p instanceof TreeNode) e = ((TreeNode<K,V>)p).putTreeVal(this, tab, hash, key, value);
如果p是一个树节点,调用putTreeVal方法
-
遍历数组这个索引下的节点
-
如果存在和要存元素key值相同的元素
如果存在,则把这个元素记录到e上,用于后续判断
-
如果不存在,则把新元素插入到链表的尾部
else { for (int binCount = 0; ; ++binCount) { if ((e = p.next) == null) { p.next = newNode(hash, key, value, null); if (binCount >= TREEIFY_THRESHOLD - 1) // -1 for 1st treeifyBin(tab, hash); break; } if (e.hash == hash && ((k = e.key) == key || (key != null && key.equals(k)))) break; p = e; } }
-
-
如果执行完上面的代码取出的元素不为空,则说明存在和要存对象key值相同的元素
if (e != null) { // existing mapping for key V oldValue = e.value; if (!onlyIfAbsent || oldValue == null) e.value = value; afterNodeAccess(e); return oldValue; }
用新的value值替换旧的value值,返回旧的value值
-
最后说明插入成功,将数组的元素数量加一,判断是否需要扩容
++modCount; if (++size > threshold) resize(); afterNodeInsertion(evict); return null; }
-
这就是put方法的全过程,此外,put方法结尾还提供了一个afterNodeAccess(e)方法,这是一个空方法,没有任何实现,它允许我们插入元素后进行一些操作
4、HashMap的取值(get方法)
get方法返回一个value值,源码如下
public V get(Object key) { Node<K,V> e; return (e = getNode(hash(key), key)) == null ? null : e.value; }
get方法通过调用getNode方法来找到相应的键值对对象,下面是getNode方法的源码
final Node<K,V> getNode(int hash, Object key) { Node<K,V>[] tab; Node<K,V> first, e; int n; K k; if ((tab = table) != null && (n = tab.length) > 0 && (first = tab[(n - 1) & hash]) != null) { if (first.hash == hash && // always check first node ((k = first.key) == key || (key != null && key.equals(k)))) return first; if ((e = first.next) != null) { if (first instanceof TreeNode) return ((TreeNode<K,V>)first).getTreeNode(hash, key); do { if (e.hash == hash && ((k = e.key) == key || (key != null && key.equals(k)))) return e; } while ((e = e.next) != null); } } return null; }
5、HashMaps数组扩容(resize方法)
因为在HashMap的构造方法里,并没有对数组进行初始化,而是在put方法里调用resize方法进行对数组的统一处理
final Node<K,V>[] resize() { Node<K,V>[] oldTab = table; int oldCap = (oldTab == null) ? 0 : oldTab.length; int oldThr = threshold; int newCap, newThr = 0;
先在resize方法里定义了一些变量来表示数组的一些信息
-
如果数组已经初始化过,分为两种情况
-
数组长度已经大于设定的最大容量了
这时,把扩容条件阈值设为整数的最大值,这样在put方法插入元素后的判断
if (++size > threshold)
就始终为false(因为size是int类型,不会大于Integer.MAX_VALUE),这样数组就不会再进行扩容 -
数组长度介于默认值和最大值之间
这时,数组将进行正常的扩容操作,数组长度和扩容条件阈值都变为原来的两倍
if (oldCap > 0) { if (oldCap >= MAXIMUM_CAPACITY) { threshold = Integer.MAX_VALUE; return oldTab; } else if ((newCap = oldCap << 1) < MAXIMUM_CAPACITY && oldCap >= DEFAULT_INITIAL_CAPACITY) newThr = oldThr << 1; // double threshold }
-
-
如果数组未初始化,且条件阈值大于零
什么时候才会发生这种情况呢?
当我们调用HashMap的非空构造方法时,在构造方法里并没有对数组进行初始化,而是计算出来扩容阈值threshold的值,这样就会出现数组长度为零,扩容条件阈值存在的情况
else if (oldThr > 0) // initial capacity was placed in threshold newCap = oldThr;
这种情况下,直接把扩容阈值作为数组的长度进行初始化
-
如果数组未初始化且条件阈值不存在
不存在的意思是没有赋过值,这时threshold的值是默认值0
这种情况最常见,一般是调用了HashMap的空构造
else { // zero initial threshold signifies using defaults newCap = DEFAULT_INITIAL_CAPACITY; newThr = (int)(DEFAULT_LOAD_FACTOR * DEFAULT_INITIAL_CAPACITY); }
这种情况下,数组长度和条件阈值都用默认值即可、
-
if (newThr == 0)?为什么还要有这个判断?
在执行
else if (oldThr > 0) newCap = oldThr;
这句后,也就是当调用HashMap的有参构造时,这里并没有为newThr赋值,因此newThr为默认值0if (newThr == 0) { float ft = (float)newCap * loadFactor; newThr = (newCap < MAXIMUM_CAPACITY && ft < (float)MAXIMUM_CAPACITY ? (int)ft : Integer.MAX_VALUE); }
这个判断就是为newThr赋值
然后把扩容后的信息更新到对象中
threshold = newThr; @SuppressWarnings({"rawtypes","unchecked"}) Node<K,V>[] newTab = (Node<K,V>[])new Node[newCap]; table = newTab;
-
扩容完毕,将原来的数据转移到新数组中
对这段代码先不做解释,谅解
二、HashMap线程不安全的原因
HashMap在java7以前,存在的线程安全问题有死循环、数据丢失、数据覆盖;在java8以后,HashMap存在的线程安全问题是数据覆盖
-
多线程问题1-->数据覆盖
前提:多个线程进行put存值时,多个线程的要存的索引位置相同,并且满足存储条件(不存在key值相同)
线程一判断完满足存储条件后挂起 -----> 线程二启动,并在在线程一要存的位置存值 ---> 线程一继续运行,在相应位置存值
这时就把线程二所存的值覆盖掉了
此外,在
if (++size > threshold)
这句代码里,++size明显是一个线程不安全的操作 -
多线程问题2-->死循环
以下是jdk7以前的HashMap数组扩容后数据迁移的源码
//数据迁移的方法,头插法添加元素 void transfer(Entry[] newTable, boolean rehash) { int newCapacity = newTable.length; //for循环中的代码,逐个遍历链表,重新计算索引位置,将老数组数据复制到新数组中去(数组不存储实际数据,所以仅仅是拷贝引用 //而已) 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); //将当前entry的next链指向新的索引位置,newTable[i]有可能为空,有可能也是个entry链,如果是entry链,直接在链 //表头部插入。 //以下三行是线程不安全的关键 e.next = newTable[i]; newTable[i] = e; e = next; } } }
会发生线程安全问题的代码主要是
e.next = newTable[i]; newTable[i] = e;
这两行.
-
多线程问题2-->数据丢失
后面会更新,敬请见谅!
这篇关于HashMap源码解析的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!
- 2025-01-11有哪些好用的家政团队管理工具?
- 2025-01-11营销人必看的GTM五个指标
- 2025-01-11办公软件在直播电商前期筹划中的应用与推荐
- 2025-01-11提升组织效率:上级管理者如何优化跨部门任务分配
- 2025-01-11酒店精细化运营背后的协同工具支持
- 2025-01-11跨境电商选品全攻略:工具使用、市场数据与选品策略
- 2025-01-11数据驱动酒店管理:在线工具的核心价值解析
- 2025-01-11cursor试用出现:Too many free trial accounts used on this machine 的解决方法
- 2025-01-11百万架构师第十四课:源码分析:Spring 源码分析:深入分析IOC那些鲜为人知的细节|JavaGuide
- 2025-01-11不得不了解的高效AI办公工具API