news 2026/2/6 5:30:56

Collections.synchronizedList、CopyOnWriteArrayList 和 synchronized对比

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Collections.synchronizedList、CopyOnWriteArrayList 和 synchronized对比

在 Java 中,Collections.synchronizedListCopyOnWriteArrayListsynchronized都可以用来解决多线程环境下的线程安全问题,但它们的性能和适用场景有所不同。以下是三者的区别和性能对比:

1.Collections.synchronizedList

Collections.synchronizedList是通过对ArrayList进行包装,使用同步锁(synchronized)来保证线程安全。

工作原理:
  • Collections.synchronizedList在每个方法(如addget等)上都加了同步锁,确保同一时间只有一个线程可以操作该集合。
优点:
  • 线程安全:适用于多线程环境。
  • 简单易用:直接通过Collections.synchronizedList包装即可。
缺点:
  • 性能较低:由于每次操作都需要获取锁,多个线程同时访问时会产生锁竞争,性能会下降。
  • 迭代器不安全:在使用迭代器遍历时,仍需要手动同步,否则可能抛出ConcurrentModificationException
示例代码:
List<String> list = Collections.synchronizedList(new ArrayList<>()); synchronized (list) { // 遍历时需要手动加锁 for (String item : list) { System.out.println(item); } }
适用场景:
  • 适合读写操作比例相对均衡的场景。
  • 适合对线程安全要求较高,但性能要求不高的场景。

2.CopyOnWriteArrayList

CopyOnWriteArrayList是一种线程安全的集合,它的实现方式是写时复制

工作原理:
  • 每次对集合进行写操作(如addremove等)时,会复制一份底层数组,修改完成后再将新的数组替换旧的数组。
  • 读操作不需要加锁,因为底层数组是只读的,读操作不会影响数据一致性。
优点:
  • 读操作性能高:读操作不需要加锁,多个线程可以同时读取数据。
  • 线程安全:写操作通过复制数组实现,避免了锁竞争。
  • 迭代器安全:迭代器是基于快照的,不会抛出ConcurrentModificationException
缺点:
  • 写操作性能较低:每次写操作都会复制整个数组,开销较大。
  • 内存占用高:频繁的写操作会导致大量的内存分配和垃圾回收。
示例代码:
List<String> list = new CopyOnWriteArrayList<>(); list.add("A"); list.add("B"); for (String item : list) { // 迭代时不需要加锁 System.out.println(item); }
适用场景:
  • 适合读多写少的场景,例如缓存、配置管理等。
  • 不适合频繁写操作的场景。

3.synchronized

synchronized是 Java 提供的关键字,用于显式地对代码块或方法加锁。

工作原理:
  • synchronized可以用来锁住代码块或方法,确保同一时间只有一个线程可以执行被锁住的代码。
优点:
  • 灵活性高:可以精确控制锁的范围。
  • 线程安全:通过锁机制保证线程安全。
缺点:
  • 性能较低:锁的粒度较大,容易导致线程阻塞和性能下降。
  • 代码复杂度高:需要手动管理锁的范围,容易出错。
示例代码:
List<String> list = new ArrayList<>(); synchronized (list) { list.add("A"); list.add("B"); }
适用场景:
  • 适合需要对代码块或方法进行精细化控制的场景。
  • 不适合需要频繁操作集合的场景,容易导致性能瓶颈。

性能对比

特性Collections.synchronizedListCopyOnWriteArrayListsynchronized
线程安全性
读操作性能中等(需要加锁)高(无锁)低(需要加锁)
写操作性能中等(需要加锁)低(写时复制,开销大)低(需要加锁)
迭代器安全性不安全(需手动加锁)安全(基于快照)不安全(需手动加锁)
适用场景读写均衡场景读多写少场景灵活控制锁的场景
内存占用高(写时复制会占用更多内存)

总结

  1. Collections.synchronizedList

    • 适合读写操作均衡的场景。
    • 性能一般,但实现简单。
  2. CopyOnWriteArrayList

    • 适合读多写少的场景。
    • 读操作性能高,但写操作性能较低,内存占用较高。
  3. synchronized

    • 适合需要精确控制锁的场景。
    • 性能较低,代码复杂度较高。

选择建议

  • 如果你的场景是读多写少,推荐使用CopyOnWriteArrayList
  • 如果你的场景是读写均衡,推荐使用Collections.synchronizedList
  • 如果需要对代码块进行精细化的线程安全控制,使用synchronized

测试代码:

public class ListTest { private static void synchronizedTest() throws InterruptedException { long start = System.currentTimeMillis(); /** * ArrayList 是非线程安全的,多个线程同时操作时可能导致索引越界。 * * Exception in thread "Thread-20" java.lang.ArrayIndexOutOfBoundsException: 10 * at java.util.ArrayList.add(ArrayList.java:465) */ List<String> list = new ArrayList<>(); int threadCount = 1000; CountDownLatch countDownLatch = new CountDownLatch(threadCount); for (int i = 0; i < threadCount; i++) { new Thread(() -> { try { Thread.sleep(500); } catch (InterruptedException e) { throw new RuntimeException(e); } // 626, 617 synchronized (list) { list.add("1"); } countDownLatch.countDown(); }).start(); } countDownLatch.await(); System.out.println("list.size():" + list.size() + " 耗时:" + (System.currentTimeMillis() - start)); for (String content : list) { System.out.println("content:" + content); } } private static void synchronizedListTest() throws InterruptedException { long start = System.currentTimeMillis(); /** * ArrayList 是非线程安全的,多个线程同时操作时可能导致索引越界。 * * Exception in thread "Thread-20" java.lang.ArrayIndexOutOfBoundsException: 10 * at java.util.ArrayList.add(ArrayList.java:465) */ List<String> list = Collections.synchronizedList(new ArrayList<>());// 622 int threadCount = 1000; CountDownLatch countDownLatch = new CountDownLatch(threadCount); for (int i = 0; i < threadCount; i++) { new Thread(() -> { try { Thread.sleep(500); } catch (InterruptedException e) { throw new RuntimeException(e); } list.add("1"); countDownLatch.countDown(); }).start(); } countDownLatch.await(); System.out.println("list.size():" + list.size() + " 耗时:" + (System.currentTimeMillis() - start)); for (String content : list) { System.out.println("content:" + content); } } private static void copyOnWriteArrayListTest() throws InterruptedException { long start = System.currentTimeMillis(); /** * ArrayList 是非线程安全的,多个线程同时操作时可能导致索引越界。 * * Exception in thread "Thread-20" java.lang.ArrayIndexOutOfBoundsException: 10 * at java.util.ArrayList.add(ArrayList.java:465) */ List<String> list = new CopyOnWriteArrayList<>();// 622 int threadCount = 1000; CountDownLatch countDownLatch = new CountDownLatch(threadCount); for (int i = 0; i < threadCount; i++) { new Thread(() -> { try { Thread.sleep(500); } catch (InterruptedException e) { throw new RuntimeException(e); } list.add("1"); countDownLatch.countDown(); }).start(); } countDownLatch.await(); System.out.println("list.size():" + list.size() + " 耗时:" + (System.currentTimeMillis() - start)); for (String content : list) { System.out.println("content:" + content); } } public static void main(String[] args) throws InterruptedException { synchronizedTest(); synchronizedListTest(); copyOnWriteArrayListTest(); }
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/2/4 21:42:54

收藏!大模型时代必看:6大热门岗位,小白程序员转岗指南

当下&#xff0c;大模型技术正以“破竹之势”渗透到各行各业&#xff0c;从智能客服到自动驾驶&#xff0c;从代码生成到科学研究&#xff0c;处处都能看到它的身影。随之而来的&#xff0c;是大模型相关岗位的爆发式增长&#xff0c;成为程序员和技术小白转型的黄金赛道。 今天…

作者头像 李华
网站建设 2026/2/5 10:43:24

内网渗透学习必备干货:12 个高价值靶场平台 + 实战避坑指南 + 操作技巧全收录!

前言 在内网渗透学习中&#xff0c;“实战靶场” 是连接理论与实操的核心桥梁 —— 它能模拟真实企业内网的拓扑结构、漏洞分布和信任关系&#xff0c;让你在合法可控的环境中练手 “跳板机横向移动”“域控提权” 等关键技能。 以下按 “新手入门→进阶实战→专项突破” 三个…

作者头像 李华
网站建设 2026/2/3 4:24:03

你不知道的Q#调试黑科技:如何让Python端实时捕获量子状态异常

第一章&#xff1a;你不知道的Q#调试黑科技&#xff1a;如何让Python端实时捕获量子状态异常在混合量子-经典计算架构中&#xff0c;Q# 与 Python 的协同调试长期面临“黑盒”困境——量子态无法直接观测&#xff0c;异常往往滞后暴露。然而&#xff0c;借助 Q# 的 DumpMachine…

作者头像 李华
网站建设 2026/2/3 21:22:55

新型僵尸网络正对路由器、摄像头等设备发起大规模DDoS攻击

新型僵尸网络正对路由器、摄像头等设备发起大规模DDoS攻击 趋势科技的一项新研究发现&#xff0c;自 2024 年底以来&#xff0c;一个新发现的物联网 &#xff08;IoT&#xff09; 僵尸网络一直利用路由器、IP 摄像头和其他连接设备等物联网设备中的漏洞&#xff0c;在全球策划…

作者头像 李华
网站建设 2026/2/5 6:34:05

看完 2025 年 IT 圈就业现状,26 届往后的考生该选计算机专业吗?

收藏&#xff01;不想35岁被淘汰&#xff1f;网络安全或许是程序员的最佳转型方向 计算机专业虽进入分化阶段&#xff0c;但网络安全人才缺口达300万&#xff0c;高端领域供不应求。高校扩招与市场需求脱节导致供需失衡&#xff0c;未来"计算机行业"的复合型人才更具…

作者头像 李华