近期项目中遇到一个不是很常见的异常ConcurrentModificationException,排查问题后,发现是由于遍历某变量时,对变量进行修改导致的,在此记录一下。
先看测试代码
import java.util.ArrayList;
public class OtherMethod {
private static ArrayList<String> myList = new ArrayList<String>();
public static void main(String[] args) {
OtherMethod.myList.add("1");
OtherMethod.myList.add("2");
for(String s:OtherMethod.myList){
System.out.println(s);
OtherMethod.myList.add("3");
}
}
}
运行后结果
1
Exception in thread "main" java.util.ConcurrentModificationException
at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:859)
at java.util.ArrayList$Itr.next(ArrayList.java:831)
at other.OtherMethod.main(OtherMethod.java:19)
可以看到,这时候程序已经出现错误信息。
这里我使用的myList变量为类变量,根据jvm中内存分配信息我们知道,类变量在jvm中是分配在方法区(Method Area)中,而类的成员变量则会分配在堆(Heap)中,会不会是由于jvm本身的设计导致的这个异常呢?重新修改代码进行验证,代码如下:
import java.util.ArrayList;
public class OtherMethod {
private ArrayList<String> myList = null;
public static void main(String[] args) {
OtherMethod method = new OtherMethod();
method.myList.add("1");
method.myList.add("2");
for(String s:method.myList){
System.out.println(s);
method.myList.add("3");
}
}
public OtherMethod(){
myList = new ArrayList<String>();
}
}
测试结果与第一次的一致,可以得出结论,该异常信息与变量类型是类变量还是成员变量无关。
实际上,这个异常产生的原因是在迭代(遍历)Vector或者Map时,对内容进行修改。此时由于修改Vector或者Map时会改变大小,导致出现这种异常信息。所以,在修改List、ArrayList、HaseMap等变量时,一定要注意,不可在迭代(遍历)过程中,否则会出现错误信息。
如果需要详细了解改问题产生的原因,可参考以下文章。
集合迭代时对集合进行修改抛ConcurrentModificationException原因的深究以及解决方案