Java集合框架非线程安全,多线程下需采取措施保证安全。1. 使用Collections.synchronizedXxx包装集合,但遍历时需手动加锁;2. 优先使用java.util.concurrent包中的ConcurrentHashMap、CopyOnWriteArrayList、BlockingQueue等高性能并发集合,内部已实现细粒度同步;3. 通过synchronized或ReentrantLock显式加锁控制对普通集合的访问;4. 减少共享状态,采用不可变集合、线程本地副本或消息传递机制。应根据场景选择合适方案:高并发读写选ConcurrentHashMap或CopyOnWriteArrayList,生产者-消费者模型用BlockingQueue,简单场景可用同步包装类,核心是避免共享非线程安全集合。

Java集合框架在多线程环境下默认是不安全的,多个线程同时读写同一个集合实例可能导致数据不一致、异常甚至程序崩溃。要保证线程安全,有多种方式可以实现,下面介绍几种常见且实用的方法。
使用同步包装类(Collections.synchronizedXxx)
Java 提供了 Collections.synchronizedList、synchronizedSet、synchronizedMap 等工具方法,可以将普通集合包装成线程安全的版本。
例如:
List
注意:虽然这些集合的方法是同步的,但遍历时仍需手动加锁,否则可能抛出 ConcurrentModificationException。
正确遍历方式:
synchronized(list) {
for (String item : list) {
// 处理元素
}
}
使用并发包中的线程安全集合
java.util.concurrent 包提供了高性能的线程安全集合,推荐在多线程场景中优先使用。
立即学习“Java免费学习笔记(深入)”;
- ConcurrentHashMap:替代 HashMap,支持高并发读写,采用分段锁或CAS机制提升性能。
- CopyOnWriteArrayList:适用于读多写少的场景,写操作会复制整个底层数组,读操作无锁。
- BlockingQueue 实现类(如 ArrayBlockingQueue、LinkedBlockingQueue):用于生产者-消费者模型,自动处理线程阻塞与唤醒。
这些集合内部已实现细粒度同步,无需额外加锁,使用更安全高效。
使用显式同步控制
如果必须使用普通集合,可以通过 synchronized 关键字或 ReentrantLock 手动控制访问。
示例:
private final List
private final Object lock = new Object();
public void add(String item) {
synchronized(lock) {
list.add(item);
}
}
public String get(int index) {
synchronized(lock) {
return list.get(index);
}
}
这种方式灵活但容易出错,需确保所有访问路径都加锁。
避免共享可变集合
最根本的解决思路是减少共享状态。可通过以下方式降低风险:
- 使用不可变集合(如 Collections.unmodifiableList)防止修改。
- 每个线程持有自己的集合副本,减少竞争。
- 通过消息传递代替共享变量(如使用队列通信)。
基本上就这些。选择哪种方式取决于具体场景:高并发读写用 ConcurrentHashMap 或 CopyOnWriteArrayList;需要阻塞操作用 BlockingQueue;简单场景可用 synchronized 包装。关键是避免在多线程下随意共享非线程安全的集合。










