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

Collections.synchronizedList是如何将List变为线程安全的

一、synchronizedList 的线程安全实现逻辑

  1. 包装类选择
    调用 Collections.synchronizedList(List<T> list) 时,会根据传入 List 是否实现 RandomAccess 接口(如 ArrayList 实现该接口,LinkedList 不实现),返回不同的包装类:
  • 实现 RandomAccess:返回 SynchronizedRandomAccessList(继承自 SynchronizedList);
  • 未实现:返回 SynchronizedList
    public static <T> List<T> synchronizedList(List<T> list) {return (list instanceof RandomAccess ?new SynchronizedRandomAccessList<>(list) :new SynchronizedList<>(list));}
  1. 核心锁机制
    线程安全的核心是 对象锁(mutex)+ synchronized 同步块
  • SynchronizedList 继承自 SynchronizedCollection,在父类中初始化 mutex 锁对象(mutex = this,即包装类实例本身);
  • getsetaddremove 等所有修改/查询方法,均通过 synchronized (mutex) 包裹,确保同一时刻只有一个线程能执行这些方法。
static class SynchronizedCollection<E> implements Collection<E>, Serializable {private static final long serialVersionUID = 3053995032091335093L;final Collection<E> c;  // Backing Collectionfinal Object mutex;     // Object on which to synchronizeSynchronizedCollection(Collection<E> c) {this.c = Objects.requireNonNull(c);mutex = this;}SynchronizedCollection(Collection<E> c, Object mutex) {this.c = Objects.requireNonNull(c);this.mutex = Objects.requireNonNull(mutex);}public int size() {synchronized (mutex) {return c.size();}}public boolean isEmpty() {synchronized (mutex) {return c.isEmpty();}}public boolean contains(Object o) {synchronized (mutex) {return c.contains(o);}}public Object[] toArray() {synchronized (mutex) {return c.toArray();}}public <T> T[] toArray(T[] a) {synchronized (mutex) {return c.toArray(a);}}public Iterator<E> iterator() {return c.iterator(); // Must be manually synched by user!}public boolean add(E e) {synchronized (mutex) {return c.add(e);}}public boolean remove(Object o) {synchronized (mutex) {return c.remove(o);}}public boolean containsAll(Collection<?> coll) {synchronized (mutex) {return c.containsAll(coll);}}public boolean addAll(Collection<? extends E> coll) {synchronized (mutex) {return c.addAll(coll);}}public boolean removeAll(Collection<?> coll) {synchronized (mutex) {return c.removeAll(coll);}}public boolean retainAll(Collection<?> coll) {synchronized (mutex) {return c.retainAll(coll);}}public void clear() {synchronized (mutex) {c.clear();}}public String toString() {synchronized (mutex) {return c.toString();}}// Override default methods in Collection@Overridepublic void forEach(Consumer<? super E> consumer) {synchronized (mutex) {c.forEach(consumer);}}@Overridepublic boolean removeIf(Predicate<? super E> filter) {synchronized (mutex) {return c.removeIf(filter);}}@Overridepublic Spliterator<E> spliterator() {return c.spliterator(); // Must be manually synched by user!}@Overridepublic Stream<E> stream() {return c.stream(); // Must be manually synched by user!}@Overridepublic Stream<E> parallelStream() {return c.parallelStream(); // Must be manually synched by user!}private void writeObject(ObjectOutputStream s) throws IOException {synchronized (mutex) {s.defaultWriteObject();}}}

二、效率低下的原因

SynchronizedList 采用 “粗粒度锁” 实现:
所有方法共用同一把 mutex 锁,无论操作是读(如 get)还是写(如 add),都会独占锁。这意味着即使多个线程仅执行读操作,也需要排队等待锁释放,无法实现“读-读并发”,导致并发效率显著低于 CopyOnWriteArrayList 等细粒度锁/无锁实现。

三、遍历需额外加锁的原因

SynchronizedList 内部的锁仅保证 单个方法调用的原子性,但无法保证 多步操作(如遍历)的原子性

  • iterator() 方法直接返回底层 List 的迭代器(未加锁),迭代器本身不具备线程安全性;
  • 若不加外部锁,可能出现“并发修改异常”(如线程 A 遍历 hasNext() 时存在元素,线程 B 立即删除该元素,线程 A 调用 next() 时会报错)。

因此,官方要求遍历 SynchronizedList 时,需在外部用 synchronized (list) 包裹迭代逻辑,确保遍历全程独占锁,避免并发修改问题。

 public Iterator<E> iterator() {return c.iterator(); // Must be manually synched by user!}

示例代码如下:

List list = Collections.synchronizedList(new ArrayList());
// 正确遍历方式
synchronized (list) {Iterator i = list.iterator(); while (i.hasNext()) {foo(i.next());}
}
http://www.xdnf.cn/news/1316107.html

相关文章:

  • Trae 辅助下的 uni-app 跨端小程序工程化开发实践分享
  • 李宏毅NLP-11-语音合成
  • 在 Element UI 的 el-table 中实现某行标红并显示删除线
  • 【PHP】Hyperf:接入 Nacos
  • Centos中内存CPU硬盘的查询
  • vscode无法检测到typescript环境解决办法
  • OpenCV 图像处理核心技术:边界填充、算术运算与滤波处理实战
  • 大模型应用发展与Agent前沿技术趋势(中)
  • JVM常用工具:jstat、jmap、jstack
  • 【Linux】IO多路复用
  • 17-线程
  • Python自学10-常用数据结构之字符串
  • Python异常、模块与包(五分钟小白从入门)
  • 文件快速复制工具,传输速度提升10倍
  • riscv中断处理软硬件流程总结
  • 【C语言强化训练16天】--从基础到进阶的蜕变之旅:Day6
  • Vue3 中的 ref、模板引用和 defineExpose 详解
  • 安卓14系统应用收不到开机广播
  • 【Java后端】Spring Boot 集成 MyBatis-Plus 全攻略
  • 大模型算法岗面试准备经验分享
  • (机器学习)监督学习 vs 非监督学习
  • 智能制造——解读37页 案例分享灯塔工厂解决方案【附全文阅读】
  • 电子电气架构 --- 自动驾驶汽车的下一步发展是什么?
  • LeetCode 分类刷题:2962. 统计最大元素出现至少 K 次的子数组
  • 零墨云A4mini打印机设置电脑通过局域网络进行打印
  • 详解flink java基础(一)
  • Flink作业执行的第一步:DataFlow graph的构建
  • nodejs 错误处理
  • Gradle快速入门学习
  • 数据结构初阶(19)外排序·文件归并排序的实现