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

并发容器(Collections)

一、并发安全问题根源

1. List(如ArrayList)
  • 问题表现:多线程同时调用addremove等方法时,可能抛出ConcurrentModificationException或导致数据不一致。
  • 根本原因
    • 非原子性操作:如add操作的流程(检查容量→扩容→插入)在多线程下可能被中断。
    • 结构性修改冲突:多线程同时修改集合结构(如扩容)导致内部数组索引混乱。
2. Set(如HashSet)
  • 问题表现:HashSet底层基于HashMap,多线程插入可能丢失数据或触发异常。
  • 根本原因
    • HashMap的线程不安全:HashSet依赖的HashMap在多线程下可能因哈希冲突或扩容导致死循环或数据丢失。
3. Map(如HashMap)
  • 问题表现:多线程并发插入或扩容时,可能造成死循环(JDK 1.7)或数据覆盖(JDK 1.8+)。
  • 根本原因
    • 哈希表扩容冲突:多线程同时触发扩容(resize)时,链表或红黑树重构过程中节点引用混乱。
    • 非原子性put操作:多线程同时插入同一哈希桶可能导致数据覆盖。

二、解决方案

1. List的线程安全实现
  • Collections.synchronizedList
    通过同步方法包装普通List,保证原子性但性能较低:

    List<String> list = Collections.synchronizedList(new ArrayList<>());
    
  • CopyOnWriteArrayList(推荐)
    写操作复制新数组,读操作无锁,适合读多写少场景:

    List<String> list = new CopyOnWriteArrayList<>();
    
2. Set的线程安全实现
  • Collections.synchronizedSet
    同步方法包装普通Set:

    Set<String> set = Collections.synchronizedSet(new HashSet<>());
    
  • CopyOnWriteArraySet(推荐)
    基于CopyOnWriteArrayList实现,适合元素少、读多写少:

    Set<String> set = new CopyOnWriteArraySet<>();
    
3. Map的线程安全实现
  • ConcurrentHashMap(推荐)

    • JDK 1.7:分段锁(Segment),降低锁粒度。
    • JDK 1.8+:CAS + synchronized锁单个哈希桶,并发性能更高。
    Map<String, String> map = new ConcurrentHashMap<>();
    
  • Collections.synchronizedMap
    同步方法包装普通Map,适用于低并发:

    Map<String, String> map = Collections.synchronizedMap(new HashMap<>());
    

三、对比与选型

集合类型非安全类安全实现方案适用场景
ListArrayListCopyOnWriteArrayList读多写少(如配置信息)
Collections.synchronizedList简单同步需求
SetHashSetCopyOnWriteArraySet元素少、读多写少
Collections.synchronizedSet低并发场景
MapHashMapConcurrentHashMap高并发读写(首选)
Collections.synchronizedMap兼容旧代码或低并发需求

四、注意事项

  1. 性能权衡
    • CopyOnWriteArrayList每次写操作复制数组,写频繁时性能差
    • ConcurrentHashMap在JDK 1.8+中优化为CAS+锁,性能接近无锁。
  2. 避免误区
    • VectorHashtable通过全表锁实现安全,但高并发下性能差,不推荐使用。
  3. 开发建议
    • 默认使用ConcurrentHashMap替代HashMapCopyOnWriteArrayList替代ArrayList
    • 若需强一致性(如金融场景),需结合显式锁(如ReentrantLock)或数据库事务。

通过选择适合的并发容器,可避免ConcurrentModificationException、数据丢失等问题,同时平衡性能与线程安全。

http://www.lryc.cn/news/2386279.html

相关文章:

  • SPA模式下的es6如何加快宿主页的显示速度
  • windows powershell 判断 进程号是否存在
  • c# 解码 encodeURIComponent
  • Spring AI:Java开发者的AI开发新利器
  • Android System UI 深度解析:从架构演进到车载 / TV 场景的全维度定制
  • Spring Cloud Sleuth与Zipkin深度整合指南:微服务链路追踪实战
  • React从基础入门到高级实战:React 基础入门 - 列表渲染与条件渲染
  • 在 stm32 中 volatile unsigned signed 分别有什么作用,分别在什么场景下使用?
  • FreeBSD14.2因为爆内存而导致Xfce4视窗被卡,桌面变黑色,只能看到鼠标在窗体中心,鼠标无反应,键盘无反应
  • k8s-NetworkPolicy
  • spring-ai 集成 mcp 之投机取巧
  • 大语言模型的完整训练周期从0到1的体系化拆解
  • 历年北京邮电大学保研上机真题
  • elementUI 中el-date-picker和el-select的样式调整
  • 《仿盒马》app开发技术分享-- 定位获取(端云一体)
  • 黑马点评--基于Redis实现共享session登录
  • Mujoco 学习系列(二)基础功能与xml使用
  • 比特授权云外壳加密支持Android 15!
  • uniapp使用sse连接后端,接收后端推过来的消息(app不支持!!)
  • 历年复旦大学保研上机真题
  • 黑马点评-实现安全秒杀优惠券(使并发一人一单,防止并发超卖)
  • 解决论文中字体未嵌入的问题
  • leetcode 131. Palindrome Partitioning
  • Android本地语音识别引擎深度对比与集成指南:Vosk vs SherpaOnnx
  • 审计报告附注救星!实现Word表格纵向求和+横向计算及其对应的智能校验
  • 人工智能数学基础实验(四):最大似然估计的-AI 模型训练与参数优化
  • 告别延迟!Ethernetip转modbustcp网关在熔炼车间监控的极速时代
  • Kotlin协程优化Android ANR问题
  • Visual Studio Code插件离线安装指南:从市场获取并手动部署
  • 构建安全AI风险识别大模型:CoT、训练集与Agent vs. Fine-Tuning对比