0

0

如何在Java中同步一个非线程安全的集合_Collections.synchronizedList全解

P粉602998670

P粉602998670

发布时间:2026-03-19 10:48:37

|

964人浏览过

|

来源于php中文网

原创

Collections.synchronizedList仅保证单方法调用线程安全,不保障复合操作原子性,如遍历中删除会抛ConcurrentModificationException;迭代器、跨方法操作(如check-then-act)及外部修改原始底层数组均需额外同步。

如何在java中同步一个非线程安全的集合_collections.synchronizedlist全解

为什么 Collections.synchronizedList 不能直接解决并发修改问题

它只是给每个方法加了 synchronized,但不保证复合操作原子性。比如遍历时一边迭代一边删元素,照样抛 ConcurrentModificationException——因为迭代器自己没锁住整个遍历过程。

常见错误现象:for (String s : list) { if (s.isEmpty()) list.remove(s); } 在多线程下大概率崩,哪怕 list 是 Collections.synchronizedList(new ArrayList())

  • 所有迭代操作(iterator()forEach()、增强 for)都需手动同步外部锁
  • list.iterator() 返回的 Iterator 本身不是线程安全的,它的 next()hasNext() 不受同步保护
  • 如果多个线程分别调用 get(i)remove(i),仍可能因 i 失效导致越界或删错

什么时候该用 Collections.synchronizedList,而不是 CopyOnWriteArrayList

它适合读多写极少、且写操作基本是追加(add())或整体替换(clear())的场景;而 CopyOnWriteArrayList 适合遍历极其频繁、写操作极少(尤其是不涉及随机位置修改)的场景。

性能差异很实在:synchronizedList 每次读都要进锁,争抢严重;CopyOnWriteArrayList 读完全无锁,但每次写都要复制整个数组,内存和 CPU 开销大。

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

SkyReels
SkyReels

SkyReels是全球首个融合3D引擎与生成式AI的AI视频创作平台

下载
  • 如果写操作包含大量 set(int, E)remove(int),别用 CopyOnWriteArrayList —— 它内部没有高效随机更新机制
  • 如果集合大小经常超过几千,又频繁写,synchronizedList 的锁粒度太粗,考虑分段锁(如 ConcurrentHashMap 改造成索引映射)或更细粒度控制
  • synchronizedList 返回的是装饰器对象,原底层数组仍可被意外暴露(比如传入 new ArrayList(decoratedList)),导致绕过同步

必须配 synchronized(list) 才安全的典型操作

任何跨多个方法调用的逻辑,只要依赖中间状态,就必须把整块代码包在同一个锁里。最典型的就是“检查后执行”(check-then-act)模式。

if (!list.contains("foo")) {
    list.add("foo"); // 非原子!两个线程可能同时通过 if,然后都 add
}

正确写法:

synchronized (list) {
    if (!list.contains("foo")) {
        list.add("foo");
    }
}
  • 遍历 + 修改:必须用 synchronized (list) { Iterator it = list.iterator(); while (it.hasNext()) { ... } }
  • 批量操作如 list.addAll(anotherList):虽然 addAll 方法本身同步了,但如果 anotherList 是非线程安全的,它可能在迭代时被其他线程改写
  • 不要用 synchronized (new Object()) 或任意新对象当锁——必须锁住 list 本体,否则不同线程锁的不是同一个对象

Collections.synchronizedList 的初始化陷阱

最容易忽略的一点:传进去的原始 list 如果已被其他代码持有引用,就等于开了后门。同步只作用于装饰器接口,不封禁底层 list 的直接访问。

错误示范:

List<String> raw = new ArrayList<>();
List<String> sync = Collections.synchronizedList(raw);
// 其他地方还在用 raw.add(...) —— 同步完全失效
  • 永远用私有 final 字段保存同步后的 list,并不暴露原始 list 引用
  • 如果必须从已有 list 构建,优先用 new ArrayList(source) 再包装,避免引用泄漏
  • 注意 synchronizedList 不会复制元素,如果元素本身可变且被共享,仍需额外同步其状态
同步集合的边界很窄:它只保方法级互斥,不保业务逻辑一致性。真正难的从来不是加锁,而是判断哪段逻辑必须锁、锁什么、锁多久。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

887

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

767

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

756

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

401

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

462

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

453

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

433

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

17171

2023.08.03

Go Web框架Gin接口开发与中间件设计实践
Go Web框架Gin接口开发与中间件设计实践

本专题围绕 Go 在 Web 后端开发中的主流框架 Gin 展开,系统讲解高性能接口开发与中间件机制设计。内容涵盖路由分组、请求绑定、参数校验、统一响应封装、日志与鉴权中间件实现,以及接口限流与异常处理策略。通过实战项目案例,帮助开发者构建结构清晰、性能优良的 Go Web 服务体系,提升接口开发效率与系统可维护性。

7

2026.03.19

热门下载

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

精品课程

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

共23课时 | 4.6万人学习

C# 教程
C# 教程

共94课时 | 11.7万人学习

Java 教程
Java 教程

共578课时 | 84.7万人学习

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

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