JDK 1.8 HashMap 扩容源码详解
发布日期:2021-05-24 16:18:25
浏览次数:48
分类:技术文章
本文共 2713 字,大约阅读时间需要 9 分钟。
作为开发人员,千万不能停留在实现功能上,一定要提升到性能方面上。这需要我们不断的实践,学习源码, 根据底层实现原理,来做出最好的操作。
就HashMap而言,一定是我们常用的集合,它有着查询速度是 O(1) 的特点,特就是快速获取键值对结果。
但是如果不去读源码,就不会知道它的默认的大小是capcity 16,负载因子是 0.75 ,在hashmap中存放 超过 caticy * 负载因子,则需要去扩容,扩容的时候,是需要消耗很多资源,因为扩容是扩了两倍,然后需要将原来的复制到新的扩容后的数组里边。
这个慢的体验就是你想要 put 的时候,如果触发了 扩容 resize() ,则需要等到扩容才能添加进去,注意,现在1.8的源码里边,扩容,就直接把原来的数据重新hash 然后给到扩容后的数组。
接下来看源码:
final Node[] resize() { Node [] oldTab = table; int oldCap = (oldTab == null) ? 0 : oldTab.length; int oldThr = threshold; int newCap, newThr = 0; if (oldCap > 0) { if (oldCap >= MAXIMUM_CAPACITY) { threshold = Integer.MAX_VALUE; return oldTab; } //newCap = oldCap << 1 这个就是把新的数组扩成原来的两倍 else if ((newCap = oldCap << 1) < MAXIMUM_CAPACITY && oldCap >= DEFAULT_INITIAL_CAPACITY) newThr = oldThr << 1; // 向左位移一位,相当于是乘以 2 } else if (oldThr > 0) // initial capacity was placed in threshold newCap = oldThr; else { // zero initial threshold signifies using defaults newCap = DEFAULT_INITIAL_CAPACITY; newThr = (int)(DEFAULT_LOAD_FACTOR * DEFAULT_INITIAL_CAPACITY); } if (newThr == 0) { float ft = (float)newCap * loadFactor; newThr = (newCap < MAXIMUM_CAPACITY && ft < (float)MAXIMUM_CAPACITY ? (int)ft : Integer.MAX_VALUE); } threshold = newThr; @SuppressWarnings({"rawtypes","unchecked"}) Node [] newTab = (Node [])new Node[newCap]; table = newTab; // 接下来就是把原来的数组里边的数据都搬到新的数组里边 if (oldTab != null) { for (int j = 0; j < oldCap; ++j) { Node e; if ((e = oldTab[j]) != null) { oldTab[j] = null; if (e.next == null) //可以看到这里是新的hash newTab[e.hash & (newCap - 1)] = e; else if (e instanceof TreeNode) ((TreeNode )e).split(this, newTab, j, oldCap); else { // preserve order Node loHead = null, loTail = null; Node hiHead = null, hiTail = null; Node next; do { next = e.next; if ((e.hash & oldCap) == 0) { if (loTail == null) loHead = e; else loTail.next = e; loTail = e; } else { if (hiTail == null) hiHead = e; else hiTail.next = e; hiTail = e; } } while ((e = next) != null); if (loTail != null) { loTail.next = null; newTab[j] = loHead; } if (hiTail != null) { hiTail.next = null; newTab[j + oldCap] = hiHead; } } } } } return newTab; }
# # 可以看到非线程安全的问题
就是在正在扩容的时候,如果有新的线程取数据,会发生错误。
转载地址:https://blog.csdn.net/star1210644725/article/details/104668391 如侵犯您的版权,请留言回复原文章的地址,我们会给您删除此文章,给您带来不便请您谅解!
发表评论
最新留言
逛到本站,mark一下
[***.202.152.39]2024年03月28日 13时17分39秒
关于作者
喝酒易醉,品茶养心,人生如梦,品茶悟道,何以解忧?唯有杜康!
-- 愿君每日到此一游!
推荐文章
git更换_git命令
2019-04-21
hp-ux 查看系统负载_Linux性能调优 | 平均负载的理解和分析
2019-04-21
elementui的tree组件页面显示不出数据_vue路由及组件
2019-04-21
python 动态tabel的数据爬取_使用requests爬取python岗位招聘数据
2021-06-24
input js number 整数_JS基础简单小结(1)
2021-06-24
二阶差分预测后数据还原公式_xgboost系列丨xgboost原理及公式推导
2021-06-24
mysql 阿里云 添加磁盘空间_rds mysql磁盘空间包含
2021-06-24
java中gui_java中GUI是什么意思?详细图解
2021-06-24
java iso 8601_如何在iOS上获得ISO 8601日期?
2021-06-24
windows8怎么下载python_win8怎么安装python
2021-06-24
linux猜数字程序,用linux实现猜数字小游戏源码
2021-06-24
linux下堆栈溢出实例,堆栈溢出在Linux上沉默?
2021-06-24
python创建nc文件_工具箱第2期 用python玩转NC
2021-06-24
拆分文件_文件拆分与合并
2019-04-21
开发优势_小程序开发优势好处有哪些
2019-04-21