CopyOnWriteArraySet适用于读多写少的并发场景,基于CopyOnWriteArrayList实现,读操作不加锁,写操作时复制整个数组以保证线程安全,避免ConcurrentModificationException,但写性能差、内存开销大,不适合高频写入或大数据量使用。

CopyOnWriteArraySet 是 Java 并发包 java.util.concurrent 中的一个线程安全的集合类,底层基于 CopyOnWriteArrayList 实现。它适用于读操作远多于写操作的并发场景。每当有修改(如 add、remove)时,都会复制底层数组,保证读操作无需加锁也能线程安全。
何时使用 CopyOnWriteArraySet
它适合以下场景:
- 集合被多个线程频繁读取,但很少修改
- 需要避免遍历时发生 ConcurrentModificationException
- 能接受写操作较慢、内存开销较大的代价
不适用于高频写入或大数据量场景,因为每次写操作都会复制整个数组,性能开销大。
基本用法示例
下面是一个简单的使用示例,展示多个线程同时读取和少量写入的情况:
立即学习“Java免费学习笔记(深入)”;
功能简介:1.用户留言功能2.用户定货功能3.定制货货功能4.定制网页样式和其实设置(比如主页)5.强大的管理功能(现在的程序都是管理功能大于应用功能:)6.管理功能支持查看订货单,留言,分页,删除等功能管理页面:login.asp管理密码:admin
在这个例子中,三个线程不断遍历集合,一个线程逐步添加元素。由于 CopyOnWriteArraySet 的特性,读线程不会抛出并发修改异常,且读操作高效。
关键特性与注意事项
理解其行为对正确使用很重要:
- 写操作加锁:add、remove 等修改操作会加锁,并复制整个底层数组
- 读操作无锁:迭代、遍历等操作不加锁,性能高
- 弱一致性迭代器:Iterator 遍历时看到的是创建迭代器时的快照,后续修改不会反映在当前遍历中
- 内存占用高:每次写操作都生成新数组,可能引发频繁 GC
- 不能存 null:add(null) 会抛出 NullPointerException
替代方案对比
根据实际需求,可考虑其他线程安全集合:
- 若需高频写入,可用 Collections.synchronizedSet(new HashSet()),配合外部同步控制
- 若数据结构固定后只读,可用 Set.copyOf(existingSet) 创建不可变集合
- 若需要排序,可考虑 ConcurrentSkipListSet
基本上就这些。CopyOnWriteArraySet 在特定读多写少的并发场景下非常有用,关键是理解它的复制机制和性能特征,避免误用于高写入频率的环境。









