Java并发修改异常

 
 

今天突然有人问我并发修改异常怎么回事,当时只记得解决方案,不是很详细的知道底层原因,花了一点点时间,看了下源码,发现其实很简单。以下是我的见解

在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();
    }


猜你喜欢

转载自blog.csdn.net/yk614294861/article/details/78656885