当前位置: 首页 > news >正文

Java高频面试之集合-18

hello啊,各位观众姥爷们!!!本baby今天来报道了!哈哈哈哈哈嗝🐶

面试官:HashMap 是线程安全的吗?多线程下会有什么问题?


HashMap 的线程安全性分析

HashMap 不是线程安全的,在多线程环境下使用可能导致数据不一致、死循环等问题。以下是详细分析:


一、多线程下的主要问题
  1. 数据覆盖(Lost Updates)

    • 场景:两个线程同时执行 put 操作,且键的哈希值相同。
    • 原因:线程 A 和 B 同时检测到桶为空,均尝试插入新节点,导致后插入的值覆盖前一个。
    • 示例
      // 线程A和B同时执行
      map.put(key1, value1); // 若两个线程的 key1 哈希到同一桶且桶为空
      map.put(key1, value2); // 最终可能只有 value2 被保留
      
  2. 链表成环(Infinite Loop)

    • JDK 1.7 问题:扩容时采用头插法,多线程并发扩容可能导致链表形成环形结构,后续 get 操作触发死循环。
    • JDK 1.8 改进:改用尾插法,但并发扩容仍可能导致数据丢失或链表断裂。
  3. Size 计算错误

    • 原因size 字段的自增操作(size++)非原子性,多线程并发修改可能导致最终值小于实际插入数。
  4. 哈希表状态不一致

    • 场景:线程 A 在扩容过程中,线程 B 并发插入数据,导致部分数据未被正确迁移到新数组。

二、问题根源
  • 非原子操作putresize 等操作涉及多个步骤(如哈希计算、链表遍历、节点插入),未加锁导致中间状态暴露。
  • 可见性问题:线程本地缓存与主内存不同步,导致读取到过期数据。

三、验证示例(JDK 1.7 链表成环)
// 线程A和B并发执行以下代码
public void unsafePut() {
    Map<Integer, Integer> map = new HashMap<>(2);
    for (int i = 0; i < 10000; i++) {
        new Thread(() -> map.put(ThreadLocalRandom.current().nextInt(), 1)).start();
    }
}
  • 结果:可能出现 CPU 占用 100%(死循环)或数据丢失。

四、解决方案
  1. 使用线程安全容器

    • ConcurrentHashMap:分段锁(JDK 1.7)或 CAS + synchronized(JDK 1.8+),保证高并发下的安全性和性能。
    • Collections.synchronizedMap:通过同步方法包装 HashMap,但性能较低。
  2. 显式同步控制

    Map<String, String> syncMap = new HashMap<>();
    // 每次操作时加锁
    synchronized (syncMap) {
        syncMap.put(key, value);
    }
    
  3. 避免共享状态

    • 线程局部存储(ThreadLocal):每个线程使用独立的 HashMap 实例。

五、性能对比
方案线程安全性能适用场景
HashMap单线程或只读多线程环境
ConcurrentHashMap中高高并发读写场景
synchronizedMap低并发场景,需兼容旧代码

🐮☺️

  • HashMap 非线程安全:多线程下可能导致数据覆盖、死循环、size 错误等问题。
  • 替代方案:优先选择 ConcurrentHashMap,或在必要时使用显式同步。
  • 设计建议:在并发编程中,始终使用线程安全的数据结构以避免潜在风险。

在这里插入图片描述

相关文章:

  • 基于SSM框架的线上甜品销售系统(源码+lw+部署文档+讲解),源码可白嫖!
  • Java并发编程 什么是分布式锁 跟其他的锁有什么区别 底层原理 实战讲解
  • 加快推进智慧水务发展,实现水务系统安全、高效运行
  • 2024年3月全国计算机等级考试真题(二级C语言)
  • 树莓派ollama docker报错尝试网上方法
  • 悟空crm v12安装好后出现 网络错误问题(已解决)
  • sql server如何提高索引命中率
  • SOLIDEDGE 至 STL 转换:数字化设计制造的关键衔接
  • Occlum 是一个内存安全的、支持多进程的 library OS,特别适用于 Intel SGX。
  • GitHub Copilot平替:CodeGeeX 2.0实测报告
  • 牛贝跟卖系统Niubox贴牌模式有哪些功能?
  • rnn的ho的维度 (num_layers * num_directions, batchsize, hidden_size)
  • NVIDIA TensorRT 深度学习推理加速引擎详解
  • 搭建Kubernetes集群
  • @Autowired 和 @Resource 注解的区别
  • 德昂观点:如何看待MicroStrategy改名为Strategy?
  • 十六进制(Hexadecimal)简介
  • 【漫话机器学习系列】161.验证曲线(Validation Curve)
  • vscode正则表达式使用
  • Python:进程间的通信,进程的操作队列
  • 广州知名网站建设后台管理便捷/搜索引擎营销的主要方法
  • 装修设计网站排行榜前十名/近日发生的重大新闻
  • 情侣手表网站/培训心得体会300字
  • 免费的php网站模板/百度问一问官网
  • 淄博网站建设 熊掌号/b站推广网站2024mmm
  • 进口食品销售销售在那个网站做/搜索seo优化