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

Java中并发修改异常如何处理

在 Java 中,ConcurrentModificationException(并发修改异常) 是遍历集合时最常见的错误之一。它发生在迭代过程中直接修改集合结构(添加/删除元素)时,与是否多线程无关。以下是详细的处理方案:


一、异常触发场景

List<String> list = new ArrayList<>(Arrays.asList("A", "B", "C"));// 会抛出 ConcurrentModificationException
for (String s : list) {if ("B".equals(s)) {list.remove(s); // 直接调用集合的 remove()}
}

二、解决方案(单线程环境)

1. 使用 Iterator.remove()(推荐)
Iterator<String> iterator = list.iterator();
while (iterator.hasNext()) {String s = iterator.next();if ("B".equals(s)) {iterator.remove(); // 安全删除当前元素}
}
2. Java 8+ 使用 removeIf()
list.removeIf(s -> "B".equals(s)); // 一行代码解决
3. 使用 CopyOnWriteArrayList(读多写少场景)
List<String> list = new CopyOnWriteArrayList<>(Arrays.asList("A", "B", "C"));
for (String s : list) {if ("B".equals(s)) {list.remove(s); // 安全但性能较低(每次修改复制整个数组)}
}
4. 遍历时记录待删元素,遍历后统一删除
List<String> toRemove = new ArrayList<>();
for (String s : list) {if ("B".equals(s)) {toRemove.add(s);}
}
list.removeAll(toRemove);
5. 使用下标遍历(仅适用于 ArrayList 等随机访问集合)
for (int i = 0; i < list.size(); i++) {String s = list.get(i);if ("B".equals(s)) {list.remove(i);i--; // 修正索引}
}

三、解决方案(多线程环境)

1. 使用并发集合类
// 线程安全的 List
List<String> list = Collections.synchronizedList(new ArrayList<>());// 遍历时需手动同步
synchronized (list) {Iterator<String> it = list.iterator();while (it.hasNext()) {String s = it.next();// 操作元素}
}
2. 使用 java.util.concurrent 包中的集合
// 高性能并发 List
List<String> list = new CopyOnWriteArrayList<>();// 并发 Map
Map<String, String> map = new ConcurrentHashMap<>();
3. 使用锁机制
List<String> list = new ArrayList<>();
ReentrantLock lock = new ReentrantLock();// 写操作加锁
lock.lock();
try {list.add("X");
} finally {lock.unlock();
}// 遍历时加锁
lock.lock();
try {for (String s : list) { /* ... */ }
} finally {lock.unlock();
}

四、关键预防原则

  1. 禁止在遍历中直接修改集合
    使用 iterator.remove() 是唯一安全的修改方式。

  2. 多线程环境优先选并发集合
    ConcurrentHashMap > Collections.synchronizedMap()
    CopyOnWriteArrayList > Collections.synchronizedList()

  3. Java 8+ 优先使用 Stream API

    list = list.stream().filter(s -> !"B".equals(s)).collect(Collectors.toList());
    
  4. 避免在迭代中调用会修改集合的方法
    包括:add(), remove(), clear(), addAll() 等。


五、常见误区

方案问题推荐替代
Collections.synchronizedList()遍历时不同步仍会抛异常遍历时加锁或用 CopyOnWriteArrayList
for 循环删除元素ArrayList 删除后索引错位Iterator.remove() 或倒序遍历
多线程用普通集合 + 同步块易遗漏同步导致并发问题直接使用 ConcurrentHashMap

最佳实践:单线程用 Iterator.remove()removeIf(),多线程用并发集合类,Java 8+ 可配合 Stream API 实现无迭代修改。

http://www.lqws.cn/news/100783.html

相关文章:

  • React 第五十二节 Router中 useResolvedPath使用详解和注意事项示例
  • 高效易用的 MAC 版 SVN 客户端:macSvn 使用体验
  • C# winform教程(二)----button
  • 行列式详解:从定义到应用
  • C# CallerMemberName特性
  • macos常见且应该避免被覆盖的系统环境变量(避免用 USERNAME 作为你的自定义变量名)
  • 6.3 day 35
  • 【iOS】多线程基础
  • iptables常用命令
  • 014校园管理系统技术解析:构建智慧校园管理平台
  • Cursor + Claude 4:微信小程序流量主变现开发实战案例
  • 【notepad++】如何设置notepad++背景颜色?
  • 如何用 pnpm patch 给 element-plus 打补丁修复线上 bug(以 2.4.4 修复 PR#15197 为例)
  • 【学习记录】深入解析 AI 交互中的五大核心概念:Prompt、Agent、MCP、Function Calling 与 Tools
  • MyBatis实战项目测试
  • GIC v3 v4 虚拟化架构
  • C++--范围for循环详解
  • 基于大模型的慢性硬脑膜下血肿预测与诊疗系统技术方案
  • 手把手教你用Appsmith打造企业级低代码平台:从部署到性能调优实战
  • 虚拟线程与消息队列:Spring Boot 3.5 中异步架构的演进与选择
  • C++中锁和原子操作的区别及取舍
  • JavaScript性能优化实战指南
  • (25) 混沌工程测试实现
  • 【JS服务器】JETBRAINS IDEs JS服务器使用什么编译JNI
  • 新手小白使用VMware创建虚拟机练习Linux
  • 从0到1,带你走进Flink的世界
  • 腾讯云国际版和国内版账户通用吗?一样吗?为什么?
  • Nginx + Tomcat负载均衡群集
  • resolvers: [ElementPlusResolver()] 有什么用?
  • POJO,DTO,VO和Model