Java常用集合与映射的线程安全问题小结

 更新时间:2025年04月11日 10:01:29   作者:jiajia651304  
本文给大家介绍Java常用集合与映射的线程安全问题小结,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧

Java常用集合与映射的线程安全问题深度解析

一、线程安全基础认知

在并发编程环境下,当多个线程同时操作同一集合对象时,若未采取同步措施,可能导致以下典型问题:

  • 数据竞争:多个线程同时修改数据导致结果不可预测
  • 状态不一致:部分线程看到集合的中间状态
  • 内存可见性:线程本地缓存与主内存数据不同步
  • 死循环风险:特定操作引发无限循环(如JDK7的HashMap扩容)

二、典型非线程安全集合问题分析

1. ArrayList的并发陷阱

// 错误示例
List<Integer> list = new ArrayList<>();
ExecutorService pool = Executors.newFixedThreadPool(10);
for (int i = 0; i < 1000; i++) {
    pool.execute(() -> list.add(new Random().nextInt()));
}
// 运行结果可能包含:元素丢失、size值异常、数组越界异常等

问题根源

  • add()方法非原子操作:elementData[size++] = e
  • 多线程同时触发扩容导致数组拷贝覆盖
  • size变量可见性问题

2. HashMap的并发灾难

Map<String, Integer> map = new HashMap<>();
// 并发执行put操作可能导致:
// 1. JDK7及之前版本:环形链表导致CPU 100%
// 2. JDK8+版本:数据丢失或size计数错误
// 3. 迭代时ConcurrentModificationException

底层机制

  • 哈希桶结构在扩容时产生链表断裂
  • 头插法(JDK7)与尾插法(JDK8)差异
  • 没有同步机制的Entry数组操作

3. HashSet的隐藏风险

Set<Integer> set = new HashSet<>();
// 本质是HashMap的包装类,所有线程安全问题与HashMap一致
// add()方法并发调用时可能产生元素丢失

三、线程安全解决方案对比

1. 同步包装方案

// 使用Collections工具类
List<String> syncList = Collections.synchronizedList(new ArrayList<>());
Map<String, Object> syncMap = Collections.synchronizedMap(new HashMap<>());
// 特征:
// 1. 所有方法使用synchronized同步块
// 2. 迭代器需要手动同步
// 3. 锁粒度大,性能较差

2. 传统线程安全集合

// Vector/Hashtable方案
Vector<String> vector = new Vector<>(); 
Hashtable<String, Integer> table = new Hashtable<>();
// 缺点:
// 1. 全表锁导致吞吐量低
// 2. 已逐渐被并发容器取代

3. 现代并发容器(java.util.concurrent包)

3.1 CopyOnWriteArrayList

List<String> cowList = new CopyOnWriteArrayList<>();
// 实现原理:
// 1. 写操作时复制新数组
// 2. 最终一致性保证
// 适用场景:读多写少(如白名单配置)

3.2 ConcurrentHashMap

Map<String, Object> concurrentMap = new ConcurrentHashMap<>();
// JDK8+实现特点:
// 1. 分段锁升级为CAS+synchronized
// 2. 节点锁粒度(锁单个哈希桶)
// 3. 支持并发度设置

3.3 ConcurrentSkipListMap

NavigableMap<String, Integer> skipMap = new ConcurrentSkipListMap<>();
// 特征:
// 1. 基于跳表实现的有序Map
// 2. 无锁读取,写入使用CAS

四、并发容器实现原理剖析

1. CopyOnWriteArrayList写时复制机制

public boolean add(E e) {
    final ReentrantLock lock = this.lock;
    lock.lock();
    try {
        Object[] elements = getArray();
        int len = elements.length;
        Object[] newElements = Arrays.copyOf(elements, len + 1);
        newElements[len] = e;
        setArray(newElements);
        return true;
    } finally {
        lock.unlock();
    }
}

2. ConcurrentHashMap并发控制

JDK8关键实现

  • 哈希桶数组+链表/红黑树
  • CAS操作实现无锁化读取
  • synchronized锁单个节点
  • size计算采用LongAdder机制

3. 并发队列实现对比

队列类型锁机制适用场景
ConcurrentLinkedQueueCAS无锁高并发生产者消费者模式
LinkedBlockingQueueReentrantLock双锁有界阻塞队列
ArrayBlockingQueue单ReentrantLock固定容量队列

五、最佳实践与注意事项

1. 选型决策指南

  • 读多写少:CopyOnWrite系列
  • 高并发写入:ConcurrentHashMap
  • 强一致性需求:同步包装类+手动锁
  • 有序性要求:ConcurrentSkipListMap

2. 常见误区规避

  • 错误认知:认为Collections.synchronizedXXX比并发容器更安全
  • 迭代器问题:未对同步集合的迭代器加锁
  • 复合操作漏洞:即使使用线程安全集合,多个操作仍需同步
// 错误示例:即使使用ConcurrentHashMap仍需同步
if (!map.containsKey(key)) {
    map.put(key, value); // 非原子操作
}
// 正确写法:
map.putIfAbsent(key, value);

3. 性能优化建议

  • 预估ConcurrentHashMap初始容量减少扩容
  • 避免在CopyOnWriteArrayList中使用超大数组
  • 合理设置并发级别(ConcurrentHashMap构造函数)
  • 使用批量操作方法(如putAll)

六、高级话题扩展

1. 弱一致性迭代器

  • ConcurrentHashMap的迭代器反映创建时的状态
  • 不保证迭代过程中数据变化可见

2. 原子复合操作

// 使用merge方法实现原子计数
ConcurrentHashMap<String, Long> counterMap = new ConcurrentHashMap<>();
counterMap.merge("key", 1L, Long::sum);

3. 分段锁的演进

  • JDK7的Segment分段锁(默认16段)
  • JDK8的Node粒度锁(锁单个哈希桶)

总结与建议

  • 严格区分场景:根据读写比例、一致性要求选择容器
  • 理解实现原理:避免误用并发容器特性
  • 组合使用锁机制:必要时搭配ReentrantLock使用
  • 监控工具辅助:使用JConsole观察容器争用情况

开发者应当建立以下意识:

  • 没有绝对线程安全的容器,只有相对安全的操作方式
  • 并发问题往往在高压场景下暴露
  • 充分测试是验证线程安全性的必要手段

通过合理选择并发容器并遵循最佳实践,可以显著降低多线程环境下的集合操作风险,构建高性能高可靠的Java应用系统。

到此这篇关于Java常用集合与映射的线程安全的文章就介绍到这了,更多相关java集合与映射内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Spring @Primary作用和实现原理详解

    Spring @Primary作用和实现原理详解

    今天分享一下Spring中的@Primary注解,Primary的意思是主要的,我们在使用spring的时候,难免会定义多个类型相同的bean,这时候如果不采取一些方法,那么是无法正常使用bean的,所以本就给大家介绍Spring @Primary的作用和实现原理
    2023-07-07
  • spring boot中nativeQuery的用法

    spring boot中nativeQuery的用法

    这篇文章主要介绍了spring boot中nativeQuery的作用,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-07-07
  • Java多线程之线程通信生产者消费者模式及等待唤醒机制代码详解

    Java多线程之线程通信生产者消费者模式及等待唤醒机制代码详解

    这篇文章主要介绍了Java多线程之线程通信生产者消费者模式及等待唤醒机制代码详解,具有一定参考价值,需要的朋友可以了解下。
    2017-10-10
  • 基于SpringBoot + Android实现登录功能

    基于SpringBoot + Android实现登录功能

    在移动互联网的今天,许多应用需要通过移动端实现与服务器的交互功能,其中登录是最常见且基础的一种功能,本篇博客将详细介绍如何使用 Spring Boot 和 Android 实现一个完整的登录功能,需要的朋友可以参考下
    2024-11-11
  • Java进程内缓存框架EhCache详解

    Java进程内缓存框架EhCache详解

    这篇文章主要介绍了Java进程内缓存框架EhCache,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2021-12-12
  • mybatis plus条件构造器queryWrapper、updateWrapper

    mybatis plus条件构造器queryWrapper、updateWrapper

    这篇文章主要介绍了mybatis plus条件构造器queryWrapper、updateWrapper,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-09-09
  • Java实现图片模糊效果详解

    Java实现图片模糊效果详解

    图片模糊是图像处理中的一种常见效果,它通过平均周围像素的颜色来使图像变得模糊,下面我们来看看如何使用Swing库实现图片模糊效果吧
    2025-02-02
  • Spring Boot 集成 Swagger2构建 API文档

    Spring Boot 集成 Swagger2构建 API文档

    这篇文章主要介绍了Spring Boot 集成 Swagger2构建 API文档,通过使用 Swagger,我们只需要按照它所给定的一系列规范去定义接口以及接口的相关信息,然后它就能帮我们自动生成各种格式的接口文档,方便前后端开发者进行前后端联调,下文需要的朋友可以参考一下
    2022-03-03
  • Java字符串相关类使用方法详解

    Java字符串相关类使用方法详解

    String、StringBuilder、StringBuffer还傻傻分不清,下面这篇文章主要给大家介绍了关于Java字符串相关类使用的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考下
    2024-07-07
  • Mybatis-Plus自动填充更新操作相关字段的实现

    Mybatis-Plus自动填充更新操作相关字段的实现

    数据库表中应该都要有create_time、update_time字段;那么在开发中,对于这些共有字段的处理应该要进行统一,这样就可以简化我们的开发过程。那么本文就对Mybatis-Plus中的字段自动填充进行记录
    2021-11-11

最新评论