CopyOnWriteArraySet是Java中基于写时复制机制的线程安全Set,内部使用CopyOnWriteArrayList实现,适用于读多写少场景。其添加或删除元素时复制整个数组,保证遍历不受修改影响,读操作无需加锁,性能高;但写操作开销大、内存占用高,且迭代器不反映实时变更。常用于监听器列表等对一致性要求不高的读密集场景,频繁写入时推荐ConcurrentHashMap.newKeySet()替代。

在多线程环境下,集合的线程安全是一个常见问题。Java 提供了多种方式来保证集合的并发安全,CopyOnWriteArraySet 是其中一种基于写时复制(Copy-On-Write)机制实现的线程安全集合。它底层依赖于 CopyOnWriteArrayList,通过在修改操作时复制整个底层数组,来避免读写冲突,非常适合读多写少的并发场景。
什么是 CopyOnWriteArraySet?
CopyOnWriteArraySet 是 Java 集合框架中 java.util.concurrent 包下的一个线程安全的 Set 实现。它的主要特点是:
- 内部使用 CopyOnWriteArrayList 存储元素,自动去重。
- 所有修改操作(如 add、remove)都会创建一个新的数组副本,原数组保持不变。
- 读操作(如 contains、iterator)无需加锁,性能高。
- 适用于读远多于写的并发场景,例如监听器列表、配置缓存等。
如何使用 CopyOnWriteArraySet?
使用方式与普通 Set 类似,但由于其线程安全性,可以直接在多线程环境中使用,无需额外同步。
基本用法示例:
立即学习“Java免费学习笔记(深入)”;
import java.util.concurrent.CopyOnWriteArraySet;
public class ConcurrentSetExample {
private static CopyOnWriteArraySet set = new CopyOnWriteArraySet<>();
public static void main(String[] args) {
// 添加元素
set.add("A");
set.add("B");
set.add("C");
// 多线程读取
new Thread(() -> {
for (String s : set) {
System.out.println("Thread-1: " + s);
}
}).start();
new Thread(() -> {
// 写操作会复制底层数组
set.add("D");
System.out.println("Thread-2 added D");
}).start();
}
}
上述代码中,一个线程遍历集合,另一个线程添加元素,不会抛出 ConcurrentModificationException,因为迭代的是旧副本。
核心特性与注意事项
理解其行为机制有助于正确使用:
- 强一致性读取不可得:由于读操作不加锁,可能读到旧数据。适合对实时性要求不高的场景。
- 写操作开销大:每次 add 或 remove 都涉及数组复制,频繁写入会影响性能。
- 内存占用较高:多个线程同时读写时,可能临时存在多个数组副本。
- 迭代器弱一致性:迭代器基于创建时的快照,不会反映后续修改,也不会抛异常。
适用场景与替代方案
根据其特点,推荐在以下情况使用:
- 集合元素较少且变动不频繁。
- 读操作远远多于写操作。
- 需要安全的并发遍历。
如果写操作频繁,可考虑其他方案:
- Collections.synchronizedSet(new HashSet()):手动同步,但遍历时仍需外部同步。
- ConcurrentHashMap.newKeySet():Java 8+ 推荐方式,性能更好,支持高并发读写。
例如:
// 更高效的线程安全 Set 替代方案 SetconcurrentSet = ConcurrentHashMap. newKeySet(); concurrentSet.add("item");
基本上就这些。CopyOnWriteArraySet 是一种简单有效的线程安全 Set 实现,关键在于理解其“读写分离”的设计思想,合理应用于读多写少的并发环境。使用时注意性能和一致性权衡,选择最适合业务场景的集合类型。不复杂但容易忽略细节。









