0

0

在Java里如何使用CopyOnWriteArrayList实现安全遍历_Java并发集合说明

P粉602998670

P粉602998670

发布时间:2026-01-21 01:01:09

|

592人浏览过

|

来源于php中文网

原创

copyonwritearraylist核心用途是多线程下边读边写不抛concurrentmodificationexception,靠读写分离和写时复制实现遍历安全,但遍历看不到新写入元素,适用于读多写少场景。

在java里如何使用copyonwritearraylist实现安全遍历_java并发集合说明

遍历时修改不会抛 ConcurrentModificationException

这是 CopyOnWriteArrayList 最核心的用途:在多线程环境下边读边写,遍历过程完全不会因其他线程调用 add()remove() 而中断。它靠的是「读写分离 + 写时复制」——每次写操作都新建数组并复制原内容,而迭代器始终持有一个快照(即创建时的数组引用),所以遍历看到的是旧数据,但不会出错。

常见错误现象:
直接用普通 ArrayList 在多线程中边 for-eachremove(),大概率触发 ConcurrentModificationException;换成 CopyOnWriteArrayList 后异常消失,但要注意——你删掉的元素,当前正在遍历的迭代器是看不到的。

  • 适用于读多写少场景(比如监听器列表、配置白名单)
  • 写操作开销大:每次 add() / remove() 都要数组复制,时间复杂度 O(n)
  • 迭代器不支持 remove()(调用会抛 UnsupportedOperationException

iterator()listIterator() 的行为差异

两者都返回只读快照视图,但语义略有不同:iterator() 是单向前向遍历,listIterator() 支持双向及索引定位。关键点在于——它们都**不反映后续写操作**,且都不允许通过迭代器修改集合。

使用场景举例:广播事件时遍历监听器列表,同时允许其他线程动态增删监听器,此时用 iterator() 最自然;若需从末尾反向处理(如清理资源),可用 listIterator(size()),但注意它仍基于构造时刻的数组。

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

  • iterator() 返回的 Iterator 不支持 remove()
  • listIterator() 支持 hasPrevious() / previous(),但同样禁止修改
  • 所有迭代器的 size()contains() 等方法都作用于快照,不是实时状态

Collections.synchronizedList() 的关键区别

别误以为加了同步锁就等于安全遍历。Collections.synchronizedList(new ArrayList()) 仅保证每个方法原子性,但复合操作(如先 size()get(i))仍需手动同步;而遍历本身(for-each)本质是多次 get() + hasNext(),中间可能被其他线程写入打断,导致不一致或异常。

CopyOnWriteArrayList 把一致性保障下沉到数据结构内部,无需额外同步块。代价是内存占用翻倍(写时复制)、写延迟明显(新元素对当前遍历不可见)。

  • 同步包装类适合需要强实时一致性的场景(如计数器、状态机)
  • CopyOnWriteArrayList 适合「最终一致 + 遍历安全」优先的场景
  • 二者都不能解决「读取后业务逻辑依赖旧值做写入」这类竞态,仍需更高层协调

实际代码里怎么避免踩坑

最常被忽略的是「写后立即读不到」这个特性。比如添加一个监听器后马上触发事件,却没被通知到——因为当前所有活跃迭代器还卡在旧快照上。

CopyOnWriteArrayList<Runnable> listeners = new CopyOnWriteArrayList<>();
listeners.add(() -> System.out.println("A"));
// 此时启动遍历
Thread t1 = new Thread(() -> {
    for (Runnable r : listeners) r.run(); // 输出 A
});
// 另一线程新增
Thread t2 = new Thread(() -> listeners.add(() -> System.out.println("B")));
t1.start(); t2.start(); // B 不会被 t1 看到
  • 不要依赖「刚 add 的元素马上能被遍历到」
  • 避免在循环体内调用 add()remove()(虽不报错,但新元素进不了本次遍历)
  • 如果需要强一致性读写,考虑 ReentrantReadWriteLock + 普通 ArrayList,而非盲目替换

它的安全是有条件的:安全在「不崩溃」,不在「实时」。理解这点,才能决定是不是真该用它。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
treenode的用法
treenode的用法

​在计算机编程领域,TreeNode是一种常见的数据结构,通常用于构建树形结构。在不同的编程语言中,TreeNode可能有不同的实现方式和用法,通常用于表示树的节点信息。更多关于treenode相关问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

548

2023.12.01

C++ 高效算法与数据结构
C++ 高效算法与数据结构

本专题讲解 C++ 中常用算法与数据结构的实现与优化,涵盖排序算法(快速排序、归并排序)、查找算法、图算法、动态规划、贪心算法等,并结合实际案例分析如何选择最优算法来提高程序效率。通过深入理解数据结构(链表、树、堆、哈希表等),帮助开发者提升 在复杂应用中的算法设计与性能优化能力。

27

2025.12.22

深入理解算法:高效算法与数据结构专题
深入理解算法:高效算法与数据结构专题

本专题专注于算法与数据结构的核心概念,适合想深入理解并提升编程能力的开发者。专题内容包括常见数据结构的实现与应用,如数组、链表、栈、队列、哈希表、树、图等;以及高效的排序算法、搜索算法、动态规划等经典算法。通过详细的讲解与复杂度分析,帮助开发者不仅能熟练运用这些基础知识,还能在实际编程中优化性能,提高代码的执行效率。本专题适合准备面试的开发者,也适合希望提高算法思维的编程爱好者。

44

2026.01.06

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

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

763

2023.08.10

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

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

376

2025.12.24

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

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

27

2026.01.21

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

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

28

2026.01.21

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

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

103

2026.02.06

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

23

2026.03.06

热门下载

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

精品课程

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

共23课时 | 4.2万人学习

C# 教程
C# 教程

共94课时 | 10.9万人学习

Java 教程
Java 教程

共578课时 | 78.6万人学习

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

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