ConcurrentModificationException发生在遍历集合时直接修改结构,正确做法是使用Iterator.remove()、并发集合如CopyOnWriteArrayList或延迟处理删除操作。

在Java中,ConcurrentModificationException 是一种常见的运行时异常,通常发生在使用迭代器遍历集合的同时,直接通过集合本身修改其结构(如添加、删除元素),而没有使用迭代器提供的安全方法。这种行为违反了“快速失败”(fail-fast)机制,从而触发异常。
大多数Java集合类(如 ArrayList、HashMap)在设计上采用“快速失败”策略。当一个线程正在遍历集合时,如果检测到集合的结构被意外修改(modCount 变化),就会抛出 ConcurrentModificationException。
常见错误示例:
List<String> list = new ArrayList<>();
list.add("A");
list.add("B");
for (String s : list) {
if (s.equals("A")) {
list.remove(s); // ❌ 直接调用list.remove()会抛出异常
}
}
在遍历过程中删除元素时,应使用 Iterator 提供的 remove() 方法,这是唯一允许在迭代期间修改集合结构的安全方式。
立即学习“Java免费学习笔记(深入)”;
正确做法:
Iterator<String> it = list.iterator();
while (it.hasNext()) {
String s = it.next();
if (s.equals("A")) {
it.remove(); // ✅ 安全删除
}
}
注意:调用 it.remove() 前必须先调用 it.next(),否则会抛出 IllegalStateException。
如果需要在多线程环境下操作集合,或者允许多个修改与遍历同时进行,可以考虑使用 java.util.concurrent 包中的线程安全集合。
示例:
List<String> list = new CopyOnWriteArrayList<>();
list.add("A");
list.add("B");
for (String s : list) {
if (s.equals("A")) {
list.remove(s); // ✅ 在CopyOnWriteArrayList中是安全的
}
}
另一种思路是将需要删除或修改的元素先收集起来,等遍历结束后再统一处理。
示例:
List<String> toRemove = new ArrayList<>
for (String s : list) {
if (s.equals("A")) {
toRemove.add(s);
}
}
list.removeAll(toRemove); // ✅ 遍历结束后再删除
这种方式适合批量删除,逻辑清晰,但需要额外内存存储临时列表。
基本上就这些。关键是要避免在遍历过程中直接修改集合结构。根据具体场景选择使用迭代器、并发集合或延迟操作,就能有效避免 ConcurrentModificationException。不复杂但容易忽略。
以上就是如何在Java中处理ConcurrentModificationException的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号