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

Java集合之【CopyOnWrite和Collections.synchronizedList()的区别】

CopyOnWriteArrayList 介绍

什么是 CopyOnWriteArrayList

适合读多写少的场景

是一个线程安全的List实现,特点是写时复制

CopyOnWriteArrayList进行修改操作(如add,set,remove)的时候,会复制原数组的值到创建的新数组中,并且读操作的时候不加锁,写操作会加锁

CopyOnWriteArrayList 的读操作

不加锁,每次写操作都会创建并复制新的数组,所以读数据的时候不会发生冲突,所以读操作不需要加锁,这样读操作的效率就会很高

CopyOnWriteArrayList 的写操作

加锁,会复制并创建新数组,因此开销大,数据量大的时候,在同一时刻会存在两倍大小的List大小的内存占用

public E set(int index, E element) {final ReentrantLock lock = this.lock;lock.lock();try {// ...CODE...} finally {lock.unlock();}
}

CopyOnWriteArrayList 会出现读写不一致吗?

会,CopyOnWriteArrayList 使用的是弱一致性。

通过设计上的取舍

  • 换取高性能的读操作

  • 保证线程安全

  • 不保证实时一致性

因此CopyOnWriteArrayList 适合读多写少的场景

代码检测

public class CopyOnWriteDemo {public static void main(String[] args) throws InterruptedException {List<String> list = new CopyOnWriteArrayList<>();list.add("A");list.add("B");// 线程:不停读Thread reader = new Thread(() -> {for (int i = 0; i < 5; i++) {System.out.println("Reader: " + list);try {Thread.sleep(200);} catch (InterruptedException ignored) {}}});// 线程:写入数据Thread writer = new Thread(() -> {try {Thread.sleep(300);} catch (InterruptedException ignored) {}list.add("C");System.out.println("Writer added C");});reader.start();writer.start();reader.join();writer.join();}
}

join():让主线程等待子线程,避免出现数据不一致,或主线程结束了子线程才结束

输出结果:

Reader: [A, B]
Writer added C
Reader: [A, B, C]
Reader: [A, B, C]
Reader: [A, B, C]
Reader: [A, B, C]

显然:

写操作对应的数组是“新”数组

读操作对应的数组是“旧”数组

写完成之后就将旧数组替换成新数组

Collections.synchronizedList() 介绍

Collections的一个包装方法,可以将List转换为线程安全的版本,对每个方法(set,get,add,remove) 都会添加一个sychronized关键字,进行同步锁,从而保证线程安全

适用场景

需要将ArrayList<>()快速转换成为线程安全的版本,以及是在简单将List转换为线程安全版本临时使用的场景

List<String> list = Collections.synchronizedList(new ArrayList<>());

缺点

缺点就是不适用于高并发的场景,因为每个操作都会使用同一个sychronized同步锁

源码所示:

 public E get(int index) {synchronized (mutex) {return list.get(index);}}
public E set(int index, E element) {synchronized (mutex) {return list.set(index, element);}
}
public void add(int index, E element) {synchronized (mutex) {list.add(index, element);}
}
public E remove(int index) {synchronized (mutex) {return list.remove(index);}
}

mutex: 一般指的是this,也就是说所有的读写都是用同一把锁

源码:

final Collection<E> c;  // Backing Collection
final Object mutex;     // Object on which to synchronizeSynchronizedCollection(Collection<E> c) {this.c = Objects.requireNonNull(c);mutex = this;
}

所以Collections.synchronizedList()是不适合在高并发场景下使用的

总结

特性 synchronizedList CopyOnWriteArrayList
线程安全方式 synchronized 锁 写时复制(读无锁)
读操作 加锁(慢) 无锁(快)
写操作 加锁(较快) 复制数组(很慢)
适用场景 写多读少 读多写少
迭代器 不安全,需手动 synchronized 安全,不会抛 CME
http://www.jsqmd.com/news/43069/

相关文章:

  • 20232324 2024-2025-1 《网络与系统攻防技术》实验六实验报告
  • Python调用C++代码
  • 复杂状态与数据流管理:分布式定时任务系统的设计
  • 【第6章 字符串】Python 字符串常用操作完全教程(含代码演示)
  • DAG-有向无环图-拓扑排序
  • MySQL EXPLAIN中的key_len:精准掌握索引使用情况
  • 1090 : 分解因数 25-11-17
  • NOIP 模拟赛 9
  • Sora 2 Cameo多角色上传+Remix二创功能API接入教程,史低0.08/条
  • info linux
  • AWS云服务深度集成
  • httpd linux 启动
  • 浅谈 Manacher
  • 第28天(简单题中等题 二分查找)
  • 基于MIMO系统的SCMA稀疏码多址接入和MPA消息传递算法matlab仿真
  • Node.js服务稳定性保障:从热更新到高可用体系
  • 一次尝试,3个小时90元的主机游玩和F1电影
  • NOIP 模拟赛 8
  • 静态路由的配置
  • 读书笔记:“外部表”的进阶使用,它主要解决了三个核心问题:如何切换文件、多用户怎么办,以及一个非常酷的玩法——把系统命令变成表。
  • [CF 2166D] Marble Council
  • DP 复习
  • 一段话 UOJ
  • PG系列:在 ​​psql​​ 客户端中定义参数与动态赋值
  • CF1375G Tree Modification 题解
  • AI评价11月17号
  • 避雷:aicodemirror.com --- 酒干倘卖无
  • 9-线性学习
  • AT AGC003 题解
  • Oracle故障处理:aix 5.3 ml6安装10.2.0.1 rac报错