今天突然有人问我并发修改异常怎么回事,当时只记得解决方案,不是很详细的知道底层原因,花了一点点时间,看了下源码,发现其实很简单。以下是我的见解
在Java开发中Exception in thread "main" java.util.ConcurrentModificationException, 这是一个并发修改异常,主要原因是迭代器遍历元素的时候,通过集合是不能修改元素的
假设我们要实现这样一个例子: 判断集合里面有没有"world"这个元素,如果有,就添加一个"javaee"元素
出现异常的代码如下:
import java.util.ArrayList;
import java.util.Iterator;
public class Test {
public static void main(String[] args) {
ArrayList<String> array = new ArrayList<String>();
// 创建并添加元素
array.add("hello");
array.add("world");
array.add("java");
Iterator it = array.iterator();
while (it.hasNext()) {
String s = (String) it.next();
if ("world".equals(s)) {
array.add("javaee");
}
}
}
}
1.异常解释
- ConcurrentModificationException:当方法检测到对象的并发修改,但不允许这种修改时,抛出此异常。
- 产生的原因:
迭代器是依赖于集合而存在的,在判断成功后,集合的中新添加了元素,而迭代器却不知道,所以就报错了,这个错叫并发修改异常。 简单描述就是:迭代器遍历元素的时候,通过集合是不能修改元素的。
- 如何解决呢?
A:迭代器迭代元素,迭代器修改元素 B:集合遍历元素,集合修改元素(普通for)
2.下面用两种方法去解决这个异常
import java.util.ArrayList;
public class Test {
public static void main(String[] args) {
ArrayList<String> array = new ArrayList<String>();
// 创建并添加元素
array.add("hello");
array.add("world");
array.add("java");
// 方式1:迭代器迭代元素,迭代器修改元素
// 而Iterator迭代器却没有添加功能,所以我们使用其子接口ListIterator
// ListIterator lit = array.listIterator();
// while (lit.hasNext()) {
// String s = (String) lit.next();
// if ("world".equals(s)) {
// lit.add("javaee");
// }
// }
// System.out.println("list1:" + array);
// 方式2:集合遍历元素,集合修改元素(普通for)
for (int x = 0; x < array.size(); x++) {
String s = (String) array.get(x);
if ("world".equals(s)) {
array.add("javaee");
}
}
System.out.println("list2:" + array);
// 方式3:增强for循环
// 增强for循环写的话会报同样的错误,因为它本身就是用来替代迭代器的
// for (String s : array) {
// if ("world".equals(s)) {
// array.add("javaee");
// }
// }
// System.out.println("list3:" + array);
}
}
提示:集合有个属性modCount,每当对集合修改一次,modCount的值就会自增,,迭代器在遍历的时候,底层会new一个Itr的内部类,初始化了一个变量记录了集合的modCount值,
此变量名是expectedModCount。迭代器在调用next()方法时候会去比较这两个值是否相等,不相等则抛出异常
final void checkForComodification() { if (modCount != expectedModCount) throw new ConcurrentModificationException(); }