java中在多线程的情况下安全的修改list(常见解决方案)

 更新时间:2025年09月04日 09:27:39   作者:藤原とラふ店丶  
本文给大家介绍java中在多线程的情况下安全的修改list的常见解决方案及实现方式,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧

在Java中,ArrayListLinkedList等常见List实现类不是线程安全的(非同步)。当多个线程同时对其进行修改(如addremove)或读写操作时,可能会导致数据不一致、ConcurrentModificationException(并发修改异常)等问题。

要在多线程环境下安全地修改List,需通过线程安全的容器同步机制保证操作的原子性和可见性。以下是常用解决方案及实现方式:

一、使用线程安全的List实现类

Java提供了几种线程安全的List实现,可直接替换非线程安全的List,无需手动处理同步。

1.Vector(古老实现,不推荐)

Vector是Java早期的线程安全List实现,其所有方法都被synchronized修饰(同步方法),保证线程安全。
缺点:同步粒度太粗(整个方法加锁),多线程并发效率低,且功能上被更优的方案替代,不推荐在新代码中使用

// Vector是线程安全的,但性能较差
List<String> vector = new Vector<>();
// 多线程可安全调用add/remove等方法
vector.add("A");
vector.remove(0);

2.Collections.synchronizedList()(包装同步,推荐基础场景)

Collections工具类的synchronizedList()方法可将任意非线程安全的List包装为线程安全的List。其原理是对所有方法添加同步锁(使用synchronized块),保证同一时刻只有一个线程能操作List

使用方式

// 1. 创建非线程安全的List(如ArrayList)
List<String> unsafeList = new ArrayList<>();
// 2. 包装为线程安全的List
List<String> safeList = Collections.synchronizedList(unsafeList);
// 多线程环境下可安全操作
// 线程1:添加元素
new Thread(() -> {
    safeList.add("A");
}).start();
// 线程2:删除元素
new Thread(() -> {
    if (!safeList.isEmpty()) {
        safeList.remove(0);
    }
}).start();

注意事项

迭代操作需手动加锁:synchronizedList返回的List在迭代时(如for-eachiterator不自动同步,需手动用synchronized块包裹,否则可能抛出ConcurrentModificationException

// 迭代时必须手动同步(锁对象为safeList本身)
synchronized (safeList) {
    for (String s : safeList) {
        System.out.println(s);
    }
}

适合读写频率均衡的场景:由于所有操作都加锁,高并发下性能一般,但实现简单,适合大多数基础场景。

3.CopyOnWriteArrayList(写时复制,推荐读多写少场景)

CopyOnWriteArrayList是Java并发包(java.util.concurrent)提供的线程安全List,其核心原理是**“写时复制”**:

  • 读操作:无需加锁,直接访问当前数组(性能极高)。
  • 写操作(addremove等):先复制一份新的数组,在新数组上修改,然后将引用指向新数组(修改时加锁,保证原子性)。

适用场景:读操作远多于写操作(如缓存、配置列表),写操作频率低但读操作需高效。

使用方式

import java.util.concurrent.CopyOnWriteArrayList;
// 初始化线程安全的CopyOnWriteArrayList
List<String> cowList = new CopyOnWriteArrayList<>();
// 多线程安全操作
// 线程1:添加元素(写操作,会复制数组)
new Thread(() -> {
    cowList.add("A");
}).start();
// 线程2:读取元素(读操作,无锁,直接访问)
new Thread(() -> {
    for (String s : cowList) {
        System.out.println(s);
    }
}).start();

优点

  • 读操作无锁,并发性能极佳(适合读多写少)。
  • 迭代时不会抛出ConcurrentModificationException(迭代的是旧数组的快照)。

缺点

  • 写操作成本高(复制数组,内存占用翻倍)。
  • 数据实时性差(读操作可能访问的是旧数组,修改后的数据需等新数组替换后才能被读取)。

二、手动同步(锁机制)

如果需要更灵活地控制同步粒度(如仅对关键修改操作加锁),可使用synchronized关键字或Lock接口手动实现同步。

1. 使用synchronized块

通过synchronized锁定List对象或其他锁对象,保证同一时刻只有一个线程执行修改操作。

List<String> list = new ArrayList<>();
// 定义锁对象(也可直接用list本身作为锁)
Object lock = new Object();
// 线程1:添加元素
new Thread(() -> {
    synchronized (lock) { // 加锁
        list.add("A");
    }
}).start();
// 线程2:删除元素
new Thread(() -> {
    synchronized (lock) { // 加锁
        if (!list.isEmpty()) {
            list.remove(0);
        }
    }
}).start();

2. 使用ReentrantLock(可重入锁)

java.util.concurrent.locks.ReentrantLock提供比synchronized更灵活的锁控制(如超时锁、公平锁),适合复杂场景。

import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
List<String> list = new ArrayList<>();
// 创建锁对象(可指定为公平锁,按请求顺序获取锁)
Lock lock = new ReentrantLock(true);
// 线程1:添加元素
new Thread(() -> {
    lock.lock(); // 加锁
    try {
        list.add("A");
    } finally {
        lock.unlock(); // 必须在finally中释放锁,避免死锁
    }
}).start();
// 线程2:删除元素
new Thread(() -> {
    lock.lock(); // 加锁
    try {
        if (!list.isEmpty()) {
            list.remove(0);
        }
    } finally {
        lock.unlock();
    }
}).start();

三、注意事项

  • 复合操作的原子性
    • 即使使用线程安全的List复合操作(如“先判断再修改”)仍需额外同步。例如:
// 错误示例:contains和add是两个独立操作,可能被其他线程打断
if (!safeList.contains("A")) { 
    safeList.add("A"); // 可能重复添加
}
// 正确:用同步块保证复合操作原子性
synchronized (safeList) {
    if (!safeList.contains("A")) {
        safeList.add("A");
    }
}
  • 迭代器的线程安全
    • synchronizedList的迭代器需手动同步(见上文)。
    • CopyOnWriteArrayList的迭代器是“快照迭代器”,不支持removeadd等修改操作(会抛UnsupportedOperationException),只能遍历。
  • 性能权衡
    • 读多写少:优先CopyOnWriteArrayList(读无锁)。
    • 读写均衡或写操作频繁:优先Collections.synchronizedList()或手动锁(避免CopyOnWriteArrayList的复制开销)。
    • 避免使用Vector(性能差,已过时)。

总结

多线程安全修改List的核心是保证操作的原子性和可见性,常用方案对比:

方案原理优点缺点适用场景
Vector同步方法简单直接性能差,同步粒度粗兼容旧代码(不推荐新用)
synchronizedList同步块包装适配所有List,实现简单所有操作加锁,并发性能一般读写均衡的基础场景
CopyOnWriteArrayList写时复制读操作无锁,性能极佳写操作成本高,数据实时性差读多写少(如缓存、配置)
手动锁(synchronized/Lock自定义同步粒度灵活控制锁范围需手动处理锁释放,易出错复杂场景(如复合操作)

根据实际业务的读写频率和复杂度选择合适方案即可。

到此这篇关于java中在多线程的情况下安全的修改list(常见解决方案)的文章就介绍到这了,更多相关java多线程修改list内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Maven Spring jar包启动报错问题解决方案

    Maven Spring jar包启动报错问题解决方案

    maven 编译jar包,放在linux服务器启动不起来,提示:xxxx-0.0.1-SNAPSHOT.jar中没有主清单属性,接下来通过本文给大家分享问题原因及解决方案,感兴趣的朋友跟随小编一起看看吧
    2023-10-10
  • 详解SpringMVC 基础教程 简单入门实例

    详解SpringMVC 基础教程 简单入门实例

    这篇文章主要介绍了详解SpringMVC 基础教程 简单入门实例,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-12-12
  • SpringBoot配置连接两个或多个数据库的实现

    SpringBoot配置连接两个或多个数据库的实现

    本文主要介绍了SpringBoot配置连接两个或多个数据库的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-05-05
  • Spring Security Oauth2.0 实现短信验证码登录示例

    Spring Security Oauth2.0 实现短信验证码登录示例

    本篇文章主要介绍了Spring Security Oauth2.0 实现短信验证码登录示例,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-01-01
  • spring-session自定义序列化方式

    spring-session自定义序列化方式

    这篇文章主要介绍了spring-session自定义序列化方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-12-12
  • 在Spring Data JPA中引入Querydsl的实现方式

    在Spring Data JPA中引入Querydsl的实现方式

    这篇文章主要介绍了在Spring Data JPA中引入Querydsl的实现方式,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-01-01
  • SpringBoot中@Import注解的使用方式

    SpringBoot中@Import注解的使用方式

    这篇文章主要介绍了SpringBoot中@Import注解的使用方式,文章围绕主题展开详细的内容介绍,具有一定的参考价值,需要的小伙伴可以参考一下
    2022-05-05
  • Java实现彩色图片转换为灰度图片的示例代码

    Java实现彩色图片转换为灰度图片的示例代码

    将彩色图片转换为灰度图片是图像处理中的常见操作,通常用于简化图像、增强对比度、或者进行后续的图像分析,本项目的目标是通过Java实现将彩色图片转换为灰度图片,需要的朋友可以参考下
    2025-02-02
  • mybatis映射器配置小结

    mybatis映射器配置小结

    本文详解MyBatis映射器配置,重点讲解字段映射的三种解决方案(别名、自动驼峰映射、resultMap),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2025-09-09
  • springboot开启mybatis驼峰命名自动映射的三种方式

    springboot开启mybatis驼峰命名自动映射的三种方式

    这篇文章给大家总结springboot开启mybatis驼峰命名自动映射的三种方式,文章并通过代码示例给大家介绍的非常详细,具有一定的参考价值,需要的朋友可以参考下
    2024-02-02

最新评论