在Java中如何使用Collections.synchronizedList实现线程安全集合_线程安全集合操作说明

使用 Collections.synchronizedList 可将普通列表包装为线程安全版本,适用于读多写少场景,但遍历等复合操作需手动加锁,且性能低于并发包中的集合类。

在多线程环境下操作集合时,普通的 ArrayList、LinkedList 等非同步集合类是不安全的,可能导致数据不一致或抛出异常。Java 提供了 Collections.synchronizedList 方法来将普通列表包装成线程安全的版本,适用于需要共享访问列表的场景。

使用 Collections.synchronizedList 包装列表

该方法接收一个 List 实例,并返回一个线程安全的同步视图。所有对列表的修改和读取操作都会被自动同步。

示例代码:

List list = new ArrayList();
List syncList = Collections.synchronizedList(list);

此时 syncList 就是一个线程安全的列表,多个线程可以安全地调用 add、remove、get 等方法。

遍历时仍需手动同步

虽然方法调用是同步的,但复合操作(如迭代)仍然需要外部加锁,否则可能引发并发问题。

正确遍历方式:

synchronized (syncList) {
  for (String item : syncList) {
    System.out.println(item);
  }
}

如果不加 synchronized 块,即使使用了 synchronizedList,也可能在遍历过程中出现 ConcurrentModificationException 或读到不一致的数据。

适用场景与注意事项

synchronizedList 适合读多写少、线程数不多的场景。它通过在每个公共方法上加 synchronized 锁实现同步,性能低于 java.util.concurrent 包中的并发集合(如 CopyOnWriteArrayList)。

  • 每次调用 get、set、add 都会获取对象锁,高并发下可能成为瓶颈
  • 不支持嵌套修改操作的原子性,例如“检查再插入”需额外同步
  • 返回的迭代器不是故障快速的(fail-fast),遍历时必须手动同步

基本上就这些。使用 Collections.synchronizedList 能快速获得一个线程安全的列表,但要注意遍历等复合操作仍需手动加锁,且性能有限,应根据实际场景选择更合适的并发集合工具。