0

0

在Java中如何为集合添加线程安全机制_Java集合线程安全设计说明

P粉602998670

P粉602998670

发布时间:2026-02-22 18:18:46

|

939人浏览过

|

来源于php中文网

原创

collections.synchronizedlist仅保证单操作原子性,复合操作需手动加锁;copyonwritearraylist适合读多写少场景;concurrenthashmap迭代器弱一致;自定义集合应使用私有锁对象而非synchronized方法。

在java中如何为集合添加线程安全机制_java集合线程安全设计说明

直接用 Collections.synchronizedList 等包装器不够安全

很多人以为调用 Collections.synchronizedList(new ArrayList()) 就万事大吉,其实只是给单个操作加了锁。遍历、条件判断等复合操作仍可能出错——比如 if (!list.isEmpty()) list.remove(0);,两步之间其他线程可能已修改集合,导致 IndexOutOfBoundsException 或逻辑错误。

实操建议:

  • 对简单读写场景(如仅做 add/get),包装器可满足基本同步需求
  • 涉及迭代、批量操作或“检查-执行”逻辑时,必须手动加锁,用 synchronized (list) { ... } 包裹整个临界区
  • 注意:包装器返回的集合,其 iterator() 方法返回的迭代器**不是线程安全的**,不能在多线程中边遍历边修改

CopyOnWriteArrayList 适合读多写少,但别滥用

它通过每次写操作复制整个数组实现线程安全,读操作完全无锁。但代价明显:内存占用高、写性能差、迭代器看到的是快照,无法反映后续写入。

适用场景:

立即学习Java免费学习笔记(深入)”;

AI Home Tab
AI Home Tab

把你喜欢的AI放到首页

下载
  • 监听器列表(如事件回调注册表)——添加/删除极少,遍历极频繁
  • 配置项缓存、白名单等静态或低频变更数据

不适用场景:

  • 元素数量大(>1000)、写操作频繁(如每秒多次 add
  • 需要强一致性视图(如要求迭代器实时反映最新状态)
  • 使用 list.size() 做循环终止条件,因写操作期间 size 可能已变,但迭代器看不到变化,易漏处理

ConcurrentHashMap 是 Map 的首选,但 keySet()values() 不是实时视图

它采用分段锁(Java 8+ 改为 CAS + synchronized + 红黑树),并发读写性能远超 HashtableCollections.synchronizedMap。但要注意其“弱一致性”设计:

  • keySet()values()entrySet() 返回的集合**不支持结构修改**(调用 remove() 会抛 UnsupportedOperationException
  • 这些集合的迭代器是弱一致性的:不会抛 ConcurrentModificationException,但可能跳过刚插入的元素,也可能重复返回同一元素
  • 若需原子性批量操作(如“如果 key 不存在则 put”),优先用 computeIfAbsentmerge 等内置方法,而非先 containsKeyput

自定义集合类加锁时,别只锁方法,要锁对象实例

常见错误是给 public synchronized void add(E e) 加 synchronized,看似线程安全,但若多个线程持有不同包装实例(如两个 synchronizedList 包装同一个底层 ArrayList),锁对象不同,照样并发冲突。

更稳妥的做法:

  • 显式使用私有 final 锁对象:private final Object lock = new Object();,所有临界区都 synchronized (lock)
  • 避免锁 this 或公开对象(如集合字段),防止外部代码意外参与锁竞争
  • 若集合被多个组件共享,考虑将同步逻辑上移到业务层统一控制,比在集合内部加锁更可控

线程安全从来不是“套个壳就完事”,关键在明确操作边界、识别复合动作、匹配锁粒度——尤其当集合被跨模块传递时,谁负责同步、锁在哪一层,往往比用哪个类更重要。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

826

2023.08.22

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

183

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

121

2025.11.27

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

695

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

370

2025.12.24

java多线程相关教程合集
java多线程相关教程合集

本专题整合了java多线程相关教程,阅读专题下面的文章了解更多详细内容。

26

2026.01.21

C++多线程相关合集
C++多线程相关合集

本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

25

2026.01.21

C# 多线程与异步编程
C# 多线程与异步编程

本专题深入讲解 C# 中多线程与异步编程的核心概念与实战技巧,包括线程池管理、Task 类的使用、async/await 异步编程模式、并发控制与线程同步、死锁与竞态条件的解决方案。通过实际项目,帮助开发者掌握 如何在 C# 中构建高并发、低延迟的异步系统,提升应用性能和响应速度。

99

2026.02.06

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

1030

2026.02.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 3.8万人学习

C# 教程
C# 教程

共94课时 | 10万人学习

Java 教程
Java 教程

共578课时 | 70.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号