ConcurrentModificationException发生在遍历集合时直接修改结构,解决方法包括:使用Iterator的remove()方法安全删除;采用CopyOnWriteArrayList或ConcurrentHashMap等并发集合;遍历时收集待操作元素,结束后统一处理;避免增强for循环中调用集合增删方法。

在Java中,ConcurrentModificationException 通常发生在使用迭代器遍历集合的同时,直接通过集合本身修改其结构(如添加、删除元素),而迭代器并不知情。这种异常常见于 ArrayList、HashMap 等非线程安全的集合类。为了避免这个问题,可以采用以下几种方式。
在遍历集合时,如果需要删除元素,应使用迭代器提供的 remove() 方法,而不是集合本身的 remove() 方法。
remove() 方法是安全的,它会同步更新迭代器内部的状态。remove(),否则会抛出异常。示例:
Iterator<String> iterator = list.iterator();
while (iterator.hasNext()) {
String item = iterator.next();
if ("toRemove".equals(item)) {
iterator.remove(); // 安全
}
}在多线程环境下,推荐使用 java.util.concurrent 包中的集合类,它们设计上就避免了此类问题。
立即学习“Java免费学习笔记(深入)”;
CopyOnWriteArrayList:适用于读多写少的场景,每次修改都会创建新的副本,遍历时不会抛出异常。ConcurrentHashMap:支持高效的并发访问,其迭代器是弱一致性的,不会抛出 ConcurrentModificationException。示例:
CopyOnWriteArrayList<String> list = new CopyOnWriteArrayList<>
list.add("a"); list.add("b");
for (String s : list) {
if ("a".equals(s)) {
list.remove(s); // 允许,但不会影响当前迭代
}
}如果不能使用并发集合或迭代器删除,可以在遍历时先记录需要删除或添加的元素,等遍历结束后再进行修改。
示例:
List<String> toRemove = new ArrayList<>();
for (String item : list) {
if (needToRemove(item)) {
toRemove.add(item);
}
}
list.removeAll(toRemove);增强for循环(foreach)底层使用的是迭代器,因此在循环体内直接调用集合的 add() 或 remove() 方法会触发异常。
错误示例(会抛异常):
for (String item : list) {
if ("bad".equals(item)) {
list.remove(item); // 错误!
}
}基本上就这些。关键是在遍历集合时保持结构一致性,要么用安全的方法修改,要么延迟修改操作。理解集合的fail-fast机制有助于写出更健壮的代码。
以上就是在Java中如何避免ConcurrentModificationException的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号