java.util.ConcurrentModificationException异常

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/Dream_Ryoma/article/details/83182215

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 。

猜你喜欢

转载自blog.csdn.net/Dream_Ryoma/article/details/83182215