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 如侵犯您的版权,请留言回复原文章的地址,我们会给您删除此文章,给您带来不便请您谅解!

上一篇:计算机的计量单位以及常见的数据类型
下一篇:使用sharding 做分库分表以后,插入报错 Executing an update/delete query

发表评论

最新留言

逛到本站,mark一下
[***.202.152.39]2024年03月28日 13时17分39秒

关于作者

    喝酒易醉,品茶养心,人生如梦,品茶悟道,何以解忧?唯有杜康!
-- 愿君每日到此一游!

推荐文章

下列关于python2.x和3.x的区别说法正确_Python 2.x和Python 3.x版本有哪些区别?【面试题详解】... 2019-04-21
git更换_git命令 2019-04-21
hp-ux 查看系统负载_Linux性能调优 | 平均负载的理解和分析 2019-04-21
elementui的tree组件页面显示不出数据_vue路由及组件 2019-04-21
android hook sensor数据_最近,又有人在谈论Android的前景了!深入解析趋势及必备技术点... 2019-04-21
python 动态tabel的数据爬取_使用requests爬取python岗位招聘数据 2021-06-24
input js number 整数_JS基础简单小结(1) 2021-06-24
二阶差分预测后数据还原公式_xgboost系列丨xgboost原理及公式推导 2021-06-24
docker mysql服务启动失败_docker中mysql初始化及启动失败问题解决方案 2021-06-24
mysql 阿里云 添加磁盘空间_rds mysql磁盘空间包含 2021-06-24
mysql 1364 hy000_mysql SQL Error: 1364, SQLState: HY000 保存错误 2021-06-24
mysqli拓展还能用mysql_最近在学习php,其中使用了MYSQLi扩展,注意是MYSQLi不是MYSQL(因PHP7已经不支持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