Java集合线程安全实践:从ArrayList数据迁移问题到synchronizedList解决方案

一、先说结论吧

在多线程环境下选择合适的List实现至关重要。下表总结了常见List实现的特性:

集合类型 线程安全 读性能 写性能 内存占用 使用场景与注意事项
ArrayList ❌ 不安全 ⭐⭐⭐⭐⭐ 直接索引访问 ⭐⭐⭐⭐ 尾部插入高效 ⭐⭐⭐⭐ 连续存储 单线程场景 随机访问频繁,尾部插入多
LinkedList ❌ 不安全 ⭐⭐ 需遍历节点 ⭐⭐⭐⭐⭐ 任意位置插入高效 ⭐⭐ 节点开销大 频繁插入删除 头尾操作多的队列场景
Vector ✅ 安全 ⭐ 全局锁阻塞 ⭐ 全局锁阻塞 ⭐⭐⭐⭐ 连续存储 已过时 遗留代码兼容,不推荐新项目使用
synchronizedList ✅ 安全 ⭐⭐ 方法级同步 ⭐⭐ 方法级同步 ⭐⭐⭐⭐ 包装器模式 通用线程安全方案 注意:复合操作需手动同步
CopyOnWriteArrayList ✅ 安全 ⭐⭐⭐⭐⭐ 无锁读快照 ⭐ 写时复制开销大 ⭐ 多版本存储 读多写极少 迭代器安全,写操作昂贵

二、实战案例:数据迁移中的线程安全问题

最近在数据迁移任务中,我遇到了一个典型的并发问题。先看问题代码:

复制代码
public void migrateTemplate() throws InterruptedException {
    // 问题点:使用非线程安全的ArrayList
    List<CustomerRotationLog> targets = new ArrayList<>();
    
    // 从MongoDB获取待迁移数据
    List<MongodbDynamicMessage> all = mongodbDynamicMessageMapper.findAll();
    
    // 创建线程计数器,用于同步所有转换任务
    CountDownLatch latch = new CountDownLatch(all.size());

    for (MongodbDynamicMessage source : all) {
        // 使用线程池异步转换
        ThreadPoolUtil.executeSafely(() -> {
            try {
                // 数据转换
                CustomerRotationLog target = convert(source);
                targets.add(target); // 🚨 线程不安全操作!
            } finally {
                latch.countDown();
                System.out.println("处理进度:" + latch.getCount() + "/" + all.size());
            }
        });
    }
    
    // 等待所有任务完成
    latch.await();
    // 批量保存
    commonService.processInBatch(targets, baseMapper::batchInsert);
}

问题分析

这段代码表面逻辑清晰,但实际上存在严重的线程安全问题 。最终的targets.size()很可能会小于all.size()

根本原因ArrayListadd()方法是非线程安全的。在多线程并发调用时,两个主要操作:

  1. elementData[size] = element; // 存储元素

  2. size = size + 1; // 更新大小

这两个操作不是原子的,可能导致:

  • 数据覆盖:多个线程对同一位置赋值

  • 大小不一致size更新不及时或错误

  • ArrayIndexOutOfBoundsException:扩容过程中的竞争条件

三、解决方案:使用synchronizedList

针对上述问题,最简单的修复方案是使用Collections.synchronizedList()

复制代码
public void migrateTemplate() throws InterruptedException {
    // 解决方案:使用线程安全的synchronizedList
    List<CustomerRotationLog> targets = Collections.synchronizedList(new ArrayList<>());
    
    List<MongodbDynamicMessage> all = mongodbDynamicMessageMapper.findAll();
    CountDownLatch latch = new CountDownLatch(all.size());

    for (MongodbDynamicMessage source : all) {
        ThreadPoolUtil.executeSafely(() -> {
            try {
                CustomerRotationLog target = convert(source);
                targets.add(target); // ✅ 现在是线程安全的
            } finally {
                latch.countDown();
                System.out.println("处理进度:" + latch.getCount() + "/" + all.size());
            }
        });
    }
    
    latch.await();
    commonService.processInBatch(targets, baseMapper::batchInsert);
}

四、总结与最佳实践

  1. 线程安全是第一要务:在多线程环境下操作共享集合,必须选择线程安全的实现

  2. 根据场景选择方案

    • 读多写极少 :选CopyOnWriteArrayList,迭代安全,无需额外同步

    • 写多或读写均衡 :选synchronizedList,注意复合操作的手动同步

  3. 理解实现原理

    • synchronizedList:方法级同步锁,适合通用场景

    • CopyOnWriteArrayList:写时复制,适合读主导场景

  4. 避免使用遗留类Vector已过时,其设计不适合现代并发需求

相关推荐
yaoxin5211237 小时前
434. Java 日期时间 API - Period 基于日期的时间段
java·开发语言·python
noipp7 小时前
推荐题目:洛谷 P10907 [蓝桥杯 2024 国 B] 蚂蚁开会
c语言·c++·算法·编程·洛谷
何极光8 小时前
IDEA集成Maven
java·maven·intellij-idea
程序员二叉8 小时前
【JUC】ThreadLocal底层原理|内存泄漏|弱引用|跨线程传递方案
java·开发语言·面试·职场和发展·juc
程序员二叉8 小时前
【JUC】线程池全套深度详解|参数|流程|拒绝策略|调优|异常处理
java·开发语言·jvm·算法·面试·juc
老马识途2.08 小时前
在AI的帮助下理解spring的启动过程
java·前端·spring
青山木9 小时前
Hot 100 --- 轮转数组
java·数据结构·算法
徐小夕9 小时前
Loop Engineering 深度解析与实战指南(全网最全)
前端·算法·github
Qt程序员9 小时前
掌握 Linux 内核调度:从原理到实现(进程篇)
java·开发语言
code bean9 小时前
【LangChain】检索器完全指南:从向量检索到生产级 RAG 架构
java·开发语言·微服务