1. 异常触发场景:遍历集合,并对集合进行add、remove或者clear等操作的时候。
2. 示例:
package Test;
import java.util.ArrayList;
import java.util.List;
public class CollectionOperateTest {
public static void main(String[] args) {
List<Integer> list = new ArrayList<Integer>();
list.add(1);
list.add(2);
list.add(3);
list.add(4);
list.add(5);
for (Integer integer : list) {
if (integer == 3) {
list.add(6);
}
}
}
}
3. 运行结果如下:
4. 错误原因: 根据报错信息可以看到 ArrayList 中有一个内部类 Itr ,报错的具体位置是内部类 Itr 的checkForComodification方法。看 ArrayList 类内部实现:
发现这个错误是当 modCount 和 expectedModCount 不相同的时候抛出来的。查看 modCount ,可以发现这个变量是定义在 ArrayList 类内部用来记录 ArrayList 对象从创建以后被修改的次数。 expectedModCount 是定义在内部类 Itr 中的一个变量,值与 modCount 相等。而集合对象在进行add、remove、clear等方法的时候都会修改 modCount 的值,但是 expectedModCount 的值不会变化。所以在遍历集合,并且对集合进行add、remove等方法的时候就会造成这两个值不一样,就会报错。
5. 解决方法:不直接调用集合对象的add、remove等方法,而是调用集合的遍历对象 Iterator 对象的方法。
6. 示例:
package Test;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
public class CollectionOperateTest {
public static void main(String[] args) {
List<Integer> list = new ArrayList<Integer>();
list.add(1);
list.add(2);
list.add(3);
list.add(4);
list.add(5);
Iterator<Integer> iterator = list.iterator();
while (iterator.hasNext()) {
iterator.next();
iterator.remove();
}
}
}
7. 不报错原因分析:可以看到集合对象的遍历实际使用的是内部类的方式实现的。以remove方法为例:看内部类 Itr 中remove方法的实现方式:
在调用remove方法的时候,先调用checkForComodification 方法,这时 modCount 和 expectedModCount 这两个值是相同的,所以不报错。继续往下走,可以看到调用的是集合对象的remove方法,先删除这个元素,然后将 modCount 的值赋给 expectedModCount 。