Java 中 ArrayList 线程安全问题

一、Bug 场景

假设你正在开发一个多线程的 Java 应用程序,该程序中有多个线程同时访问并修改一个共享的ArrayList。例如,你有一个订单处理系统,不同的线程负责添加新订单到订单列表,同时有其他线程可能会遍历订单列表进行统计等操作。

二、代码示例

java 复制代码
import java.util.ArrayList;
import java.util.List;

public class ArrayListThreadBug {
    private static List<Integer> orderList = new ArrayList<>();

    public static void main(String[] args) {
        // 启动添加订单线程
        Thread addThread1 = new Thread(() -> {
            for (int i = 0; i < 1000; i++) {
                orderList.add(i);
            }
        });

        Thread addThread2 = new Thread(() -> {
            for (int i = 1000; i < 2000; i++) {
                orderList.add(i);
            }
        });

        // 启动遍历订单线程
        Thread traverseThread = new Thread(() -> {
            for (Integer order : orderList) {
                System.out.println(order);
            }
        });

        addThread1.start();
        addThread2.start();
        traverseThread.start();
    }
}

三、问题描述

在运行上述代码时,你可能会遇到ConcurrentModificationException异常。这是因为ArrayList不是线程安全的,当多个线程同时对其进行修改和遍历操作时,就可能出现这种问题。在遍历过程中,如果其他线程修改了列表的结构(例如添加或删除元素),就会破坏迭代器的一致性,从而抛出该异常。

四、解决方案

  1. 使用线程安全的集合类 :可以将ArrayList替换为CopyOnWriteArrayList,它在修改时会创建一个新的数组,从而保证遍历操作的线程安全性。
java 复制代码
import java.util.List;
import java.util.concurrent.CopyOnWriteArrayList;

public class CopyOnWriteArrayListExample {
    private static List<Integer> orderList = new CopyOnWriteArrayList<>();

    public static void main(String[] args) {
        // 启动添加订单线程
        Thread addThread1 = new Thread(() -> {
            for (int i = 0; i < 1000; i++) {
                orderList.add(i);
            }
        });

        Thread addThread2 = new Thread(() -> {
            for (int i = 1000; i < 2000; i++) {
                orderList.add(i);
            }
        });

        // 启动遍历订单线程
        Thread traverseThread = new Thread(() -> {
            for (Integer order : orderList) {
                System.out.println(order);
            }
        });

        addThread1.start();
        addThread2.start();
        traverseThread.start();
    }
}
  1. 使用同步机制 :通过synchronized关键字来同步对ArrayList的访问,确保同一时间只有一个线程能修改或遍历列表。
java 复制代码
import java.util.ArrayList;
import java.util.List;

public class SynchronizedArrayListExample {
    private static List<Integer> orderList = new ArrayList<>();
    private static final Object lock = new Object();

    public static void main(String[] args) {
        // 启动添加订单线程
        Thread addThread1 = new Thread(() -> {
            synchronized (lock) {
                for (int i = 0; i < 1000; i++) {
                    orderList.add(i);
                }
            }
        });

        Thread addThread2 = new Thread(() -> {
            synchronized (lock) {
                for (int i = 1000; i < 2000; i++) {
                    orderList.add(i);
                }
            }
        });

        // 启动遍历订单线程
        Thread traverseThread = new Thread(() -> {
            synchronized (lock) {
                for (Integer order : orderList) {
                    System.out.println(order);
                }
            }
        });

        addThread1.start();
        addThread2.start();
        traverseThread.start();
    }
}
相关推荐
JAVA面经实录9171 天前
Java企业级工程化·终极完整版背诵手册(无遗漏、全覆盖、面试+落地通用)
java·开发语言·面试
许彰午1 天前
CacheSQL(二):主从复制——OpLog 环形缓冲区与故障自动恢复
java·数据库·缓存
Bat U1 天前
JavaEE|多线程初阶(七)
java·开发语言
掌心向暖RPA自动化1 天前
如何获取网页某个元素在屏幕可见部分的中心坐标影刀RPA懒加载坐标定位技巧
java·javascript·自动化·rpa·影刀rpa
日取其半万世不竭1 天前
Minecraft Java版社区服务器搭建教程(Linux,适合新手)
java·linux·服务器
TeamDev1 天前
JxBrowser 9.0.0 版本发布啦!
java·前端·混合应用·jxbrowser·浏览器控件·跨平台渲染·原声输入
AI人工智能+电脑小能手1 天前
【大白话说Java面试题】【Java基础篇】第24题:Java面向对象有哪些特征
java·开发语言·后端·面试
AI人工智能+电脑小能手1 天前
【大白话说Java面试题】【Java基础篇】第25题:JDK1.8的新特性有哪些
java·开发语言·后端·面试
likerhood1 天前
SLF4J: Failed to load class “StaticLoggerBinder“ 解决
java·log4j·maven
早日退休!!!1 天前
大模型推理瓶颈七层分析模型
java·服务器·数据库