Collections.synchronizedList可将普通List转为线程安全列表,但遍历时需手动加锁,复合操作需外部同步,适用于读写均衡场景;读多写少时推荐CopyOnWriteArrayList。

在Java并发编程中,多个线程同时访问同一个集合时可能会导致数据不一致或抛出异常。为了保证线程安全,Java提供了多种并发工具类,其中 Collections.synchronizedList 是一种简单有效的方式,用于将普通的非线程安全列表转换为线程安全的列表。
什么是 Collections.synchronizedList?
Collections.synchronizedList 是 Collections 工具类中的一个静态方法,它可以将一个普通的 List 包装成一个线程安全的 List。该方法返回一个同步(synchronized)的列表,所有对该列表的读写操作都会被同步,从而避免多线程环境下的并发问题。
例如:
ListList
此时,syncList 就是一个线程安全的列表,多个线程可以安全地对其进行增删改查操作。
立即学习“Java免费学习笔记(深入)”;
使用注意事项
虽然 synchronizedList 提供了基础的线程安全支持,但在实际使用中仍有一些关键点需要注意:
- 遍历必须手动同步:即使列表是同步的,迭代操作(如 for-each 循环)也不是自动线程安全的。在遍历时,需要手动对列表对象加锁。
正确做法示例:
在现实生活中的购物过程,购物者需要先到商场,找到指定的产品柜台下,查看产品实体以及标价信息,如果产品合适,就将该产品放到购物车中,到收款处付款结算。电子商务网站通过虚拟网页的形式在计算机上摸拟了整个过程,首先电子商务设计人员将产品信息分类显示在网页上,用户查看网页上的产品信息,当用户看到了中意的产品后,可以将该产品添加到购物车,最后使用网上支付工具进行结算,而货物将由公司通过快递等方式发送给购物者
for (String item : syncList) {
System.out.println(item);
}
}
- 复合操作需额外同步:像检查再插入(check-then-put)、size + add 等操作虽然是调用同步方法,但由于不是原子操作,在多线程环境下仍可能出错。
错误示例:
if (!syncList.contains("value")) {syncList.add("value"); // 可能多个线程同时通过判断
}
这种情况下,应使用外部同步块来保证原子性:
synchronized (syncList) {if (!syncList.contains("value")) {
syncList.add("value");
}
}
与 CopyOnWriteArrayList 的对比
除了 synchronizedList,Java 并发包还提供了 CopyOnWriteArrayList,它是另一种线程安全的 List 实现。
- synchronizedList 基于互斥锁实现,每次操作都加锁,适合读写均衡或写少的场景,但高并发下性能较低。
- CopyOnWriteArrayList 使用“写时复制”机制,写操作会复制整个底层数组,适用于读多写少的场景,读操作完全无锁,性能高。
选择建议:
- 如果频繁修改列表内容,优先考虑 synchronizedList。
- 如果主要是读操作,极少写入,推荐使用 CopyOnWriteArrayList。
总结
Collections.synchronizedList 是一种快速将普通 List 转为线程安全的方法,适合在已有 List 基础上增加线程安全能力。但它并不解决所有并发问题,遍历时需手动加锁,复合操作也需额外同步控制。合理使用它,并根据实际场景权衡是否使用更高效的并发容器,才能写出高效且安全的并发代码。
基本上就这些,别忘了遍历加锁,不然还是会有问题。










