目录

  • `HashMap` 的线程不安全
  • `HashMap` 中的 `put()` 方法
    • 数据的覆盖一
    • 数据的覆盖二

HashMap 的线程不安全

HashMap 的线程不安全主要体现在下面两个方面

  • jdk 1.7 中,当并发执行扩容操作时会造成环形链和数据丢失的情况
  • jdk 1.8 中,在并发执行 put 操作时会发生数据覆盖的情况

对于 jdk 1.7HashMap 的线程不安全,暂且不谈了,我们主要看看 jdk 1.8 中的

HashMap 中的 put() 方法

put() 方法是 jdk 1.8 中的

public V put(K key, V value) {return putVal(hash(key), key, value, false, true);
}final V putVal(int hash, K key, V value, boolean onlyIfAbsent,boolean evict) {Node<K,V>[] tab; Node<K,V> p; int n, i;// 判断 table[] 是否为空,如果是空的就创建一个 table[],并获取他的长度nif ((tab = table) == null || (n = tab.length) == 0)n = (tab = resize()).length;   // 如果单链表节点 Node<K,V> p == tab[i = (n - 1) & hash]) == null,// 就直接 put 进单链表中,说明此时并没有发生 Hash 冲突if ((p = tab[i = (n - 1) & hash]) == null)tab[i] = newNode(hash, key, value, null);else {// 说明索引位置已经放入过数据了,已经在单链表处产生了Hash冲突Node<K,V> e; K k;// 判断 put 的数据和之前的数据是否重复if (p.hash == hash &&// 进行 key 的 hash 值和 key 的 equals() 和 == 比较,如果都相等,则初始化数组 Node<K,V> e((k = p.key) == key || (key != null && key.equals(k))))              e = p;// 判断是否是红黑树,如果是红黑树就直接插入树中else if (p instanceof TreeNode)e = ((TreeNode<K,V>)p).putTreeVal(this, tab, hash, key, value);else {// 如果不是红黑树,就遍历每个节点,判断单链表长度是否大于等于 7,// 如果单链表长度大于等于 7,数组的长度小于 64 时,会优先选择扩容// 如果单链表长度大于等于 7,数组的长度大于 64 时,才会选择单链表--->红黑树for (int binCount = 0; ; ++binCount) {if ((e = p.next) == null) {// 采用尾插法,在单链表中插入数据p.next = newNode(hash, key, value, null);// 如果 binCount >= 8 - 1if (binCount >= TREEIFY_THRESHOLD - 1) treeifyBin(tab, hash);break;}// 判断索引每个元素的key是否可要插入的key相同,如果相同就直接覆盖if (e.hash == hash &&((k = e.key) == key || (key != null && key.equals(k))))break;p = e;}}// 说明数组或者单链表中有相同的key,因此只需要将value覆盖,并将oldValue返回即可if (e != null) { V oldValue = e.value;if (!onlyIfAbsent || oldValue == null)e.value = value;afterNodeAccess(e);return oldValue;}}// 说明没有key相同,因此要插入一个key-value,并记录内部结构变化次数++modCount;// 判断是否扩容if (++size > threshold)resize();afterNodeInsertion(evict);return null;
}

数据的覆盖一

  • 13 行代码是判断是否出现 hash 冲突的,假设两个线程 A、B 都在进行 put 操作,并且它们 put 数据的 keyhash 值是相同的,同时它们 keyA == keyBtrue 或者 keyA.equals(keyB)true,也就是说它们 put 数据的 value 是不相同的
  • 当线程 A 执行完第 13 行代码后由于时间片耗尽导致被挂起,而线程 B 得到时间片后在该单链表处插入了元素,完成了正常的插入
  • 然后线程 A 获得时间片,由于之前已经进行了 hash 冲突的判断,所有此时不会再进行判断,而是直接进行插入覆盖,这就导致了线程 B 插入的数据被线程 A 覆盖了,从而发生了线程不安全

数据的覆盖二

  • 58 行处有个 ++size,我们这样想,还是线程 A、B,这两个线程同时进行 put 操作时,假设当前 HashMapsize 大小为 10
  • 当线程 A 执行到第 58 行代码时,从主内存中获得 size 的值为 10 后准备进行 +1 操作,但是由于时间片耗尽只好让出 CPU
  • 于是线程 B 得到 CPU 调度,还是从主内存中拿到 size 的值 10 进行 +1 操作,完成了 put 操作,并将 size = 11 写回了主内存
  • 然后线程 A 再次得到 CPU 调度,并继续执行(此时 size 的值仍为10),当执行完 put 操作后,还是将 size = 11 写了回内存。此时,线程 A、B 都执行了一次 put 操作,但是 size 的值只增加了 1,所有说还是由于数据覆盖又导致了线程不安全
// HashMap 中 size 变量
transient int size;

HashMap为什么是线程不安全的相关推荐

  1. 常见面试题:为什么HashMap不是线程安全的呢?(JDK1.7和JDK1.8角度)(看完你就能和面试官笑谈人生了)

    title: 常见面试题:为什么HashMap不是线程安全的呢?(JDK1.7和JDK1.8角度)(看完你就能和面试官笑谈人生了) tags: 面试常见题 常见面试题:为什么HashMap不是线程安全 ...

  2. Java多线程学习二十:HashMap 为什么是线程不安全的

    为什么 HashMap 是线程不安全的?而对于 HashMap,相信你一定并不陌生,HashMap 是我们平时工作和学习中用得非常非常多的一个容器,也是 Map 最主要的实现类之一,但是它自身并不具备 ...

  3. hashmap为什么是线程不安全的_HashMap 为什么线程不安全?

    1.jdk1.7中的HashMap 1.1 扩容造成死循环分析过程 1.2 扩容造成数据丢失分析过程 2.jdk1.8中HashMap 总结 前言:我们都知道HashMap是线程不安全的,在多线程环境 ...

  4. HashMap为什么是线程不安全的?

    2019独角兽企业重金招聘Python工程师标准>>> 一直以来只是知道HashMap是线程不安全的,但是到底HashMap为什么线程不安全,多线程并发的时候在什么情况下可能出现问题 ...

  5. HashMap和Hashtable 线程安全性

    HashMap和Hashtable的比较是Java面试中的常见问题,用来考验程序员是否能够正确使用集合类以及是否可以随机应变使用多种思路解决问题.HashMap的工作原理.ArrayList与Vect ...

  6. 集合类ArrayList、HashMap、HashSet线程不安全

    为了提高写入性能JDK 1.2 引入了非线程安全容器ArrayList替代Vector package Juc;import java.util.ArrayList; import java.util ...

  7. 谈谈HashMap为什么是线程不安全的?

    少年不惧岁月长,彼方尚有荣光在 推荐阅读: <面试必问-HashMap>通俗易懂搞定HashMap底层原理 我们都知道HashMap是线程不安全的,在多线程环境中不建议使用,应该使用Con ...

  8. HashMap 为什么线程不安全?

    点击上方"方志朋",选择"设为星标" 回复"666"获取新整理的面试文章 来源:developer cnblogs.com/develope ...

  9. 都说 HashMap 是线程不安全的,到底体现在哪儿?

    前言:我们都知道HashMap是线程不安全的,在多线程环境中不建议使用,但是其线程不安全主要体现在什么地方呢,本文将对该问题进行解密. 1.jdk1.7中的HashMap 在jdk1.8中对HashM ...

  10. python 线程安全链表_教你用 Python 实现 HashMap 数据结构

    <犬夜叉·镜中的梦幻城> 今天这篇文章给大家讲讲hashmap,这个号称是所有Java工程师都会的数据结构.为什么说是所有Java工程师都会呢,因为很简单,他们不会这个找不到工作.几乎所有 ...

最新文章

  1. javascript中 (function(){})();如何理解?
  2. 浅谈深度学习图像分割
  3. Unity 引擎UGUI之自定义树形菜单(TreeView)
  4. directshow怎样打开摄像头不预览只抓帧_不比不知道,一比还真有差距!四款高性价比家庭智能摄像头对比...
  5. Java HttpSessionListener监听器的使用
  6. 使用 user agent 判断微信内置浏览器版本信息
  7. C4C Product Price List的模型中和有效期相关的两个字段
  8. [css] CSS中的calc()有什么作用?
  9. Matrix使用的分析
  10. 07.配置日志的存储路径、设置日志的格式
  11. 五大常用算法(一) - 分治算法
  12. 树形控件之思维导图 Android
  13. JAVA根据时间增加1天
  14. 有人称2022年将会是DAO的元年
  15. Redis数据结构-字符串对象(SDS)
  16. 农村小伙不畏艰难,创业开装饰公司,年产值竟达上千万元?
  17. NVIDIA CUDA 高度并行处理器编程(一):CUDA简介
  18. 【BZOJ4864】【BJWC2017】神秘物质 - Splay
  19. 如何在rhel4上禁用不需要的相关服务
  20. 已有打开的与此 Command 相关联的 DataReader,必须首先将它关闭。

热门文章

  1. java jsonobject 清空_有没有办法,我可以清空整个JSONObject – java
  2. CTR预估之outbrain
  3. Kaggle Future Sales“”竞赛 XGB_model_final
  4. InvalidArchiveError(‘Error with archive D:\\Program Files\\Anaconda\\pkgs\\numpy-base-1.19.1-py36ha3
  5. linux SCP远程拷贝文件方法及not a regular file 错误解决方法
  6. 6Jordan标准型的变换与应用
  7. 持续交付2.0 pdf_便捷下载发布v7.2.0版本更新
  8. MacOS 迅速上手 Makefile 编译 C / C++ 工程
  9. indesign用于产品排班_2019年机器人行业十大新品盘点,过去一年最受关注的产品都在这...
  10. linux虚拟机安装samba服务,在虚拟机Redhat Linux下安装Samba服务器分享