当前位置: 首页 > news >正文

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(); }
http://www.cnnetsun.cn/news/115104.html

相关文章:

  • 腾讯云国际站代理商的QAPM服务能提供哪些专属服务?
  • 网安副业怎么选?漏洞挖掘、技术博客、竞赛奖金实战,哪个更适配你?
  • 量子计算验证方法:软件测试从业者的转型指南
  • 突破 Oracle/MySQL 瓶颈:金仓数据库以三重革新,筑牢业务转型 “数据底座”
  • 【学习神器】NotebookLM“播客”功能实战指南:四六级、考研党高效复习秘籍
  • 如何解决 pip install 网络报错 ERROR: No matching distribution found for requests
  • 12 Ways to Find User Account Info and Login Details in Linux
  • 紧急警告:错误的导出格式正毁掉你的量子实验成果,速查正确方式
  • 35 岁职场焦虑蔓延?为什么网络安全行业越老越值钱?
  • 内网渗透实战干货:12 个优质靶场平台精选,附避坑指南 + 实操技巧合集!
  • 新型电力系统下多分布式电源接入配电网承载力评估方法研究附Matlab代码
  • 50天学习FPGA第16天-verilog的模块与端口
  • 50天学习FPGA第15天-verilog基本概念
  • 基于Docker容器化部署Lsky Pro私有图床系统
  • GRPO不香了?小米ICPO横空出世,专治大模型“不会思考”,推理能力飙升!
  • Windows找不到xenroll.dll文件 如何下载修复?
  • 软件测试文档标准化编写指南
  • Paperzz AI:毕业论文写作的 “隐形助攻”,让学术输出告别 “抓瞎”
  • BypassAV通过Patch白文件实现Bypass,没有添加其他免杀手法
  • 鸿蒙:一个操作系统的生态远征与多行业渗透之路
  • 游戏启动缺少X3DAudio1_3.dll文件问题 下载修复
  • java毕业设计之基于数据安全的旅游民宿租赁系统源代码(java+springboot+mysql)
  • 基于SpringAI构建大模型应用
  • 黑锋科技(HeifengTech)过压过流保护开关芯片全系列技术解析
  • DVWA -SQL Injection-通关教程-完结
  • AI大模型:未来就业的吞噬者还是创造者?揭秘其对普通人工作的影响!
  • 0x3f第七天 二叉搜索树
  • 扩容U盘,资料毁灭盘
  • 数据结构学习篇(5)---顺序表和链表的区别
  • 基于Vue.js和Spring Boot的新能源汽车充电站管理系统的设计与实现文献综述