3种方式解决iterator迭代器并发修改异常

3种方式解决iterator迭代器并发修改异常ConcurrentModificationException

在使用迭代器的时候,时长会遇到 ConcurrentModificationException(并发修改异常)

这也是很多人头疼的问题

并发修改异常产生的原因

在使用迭代器迭代集合的同时,使用原集合修改元素;如果迭代器发现自己和集合不一样,就会抛出 ConcurrentModificationException 异常。

先拿出我写的小案例:

   Collection<String> list = new ArrayList<>();
    list.add("JAVA");
    list.add("Python");
    list.add("PHP");
    Iterator<String> it = list.iterator();
    while(it.hasNext()) {
        if (it.next().equals("PHP")) {
            list.add("我全都要");
        }
    }

一运行,就抛异常

Exception in thread "main" java.util.ConcurrentModificationException
	at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:909)
	at java.util.ArrayList$Itr.next(ArrayList.java:859)
	at Test.main(Test.java:13)

看看源码:

at java.util.ArrayList$Itr.next(ArrayList.java:859)

    public E next() {
        checkForComodification();
        int i = cursor;
        if (i >= size)
            throw new NoSuchElementException();
        Object[] elementData = ArrayList.this.elementData;
        if (i >= elementData.length)
            throw new ConcurrentModificationException();
        cursor = i + 1;
        return (E) elementData[lastRet = i];
    }

at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:909)

    final void checkForComodification() {
        if (modCount != expectedModCount)
            throw new ConcurrentModificationException();
    }

modCount 为 ArrayList 的类成员变量,用来记录其变化次数;而 expectedModCount 作为迭代器成员变量,则存储了 iterator 初始化时记录到的 ArrayList 中的 modCount 值。如果 modConut 和 expectedModCount 不相等,则抛出 ConcurrentModificationException 异常。

解决方法

这里我使用3种解决方案

  1. 使用列表迭代器
  2. 不使用迭代器遍历,使用普通for遍历
  3. 使用toArray
第一种:使用列表迭代器
    List<String> list = new ArrayList<>();
    list.add("JAVA");
    list.add("Python");
    list.add("PHP");
    ListIterator<String> it = list.listIterator();

    while(it.hasNext()){
        if (it.next().equals("PHP")) {
            // list.add("以上几个");  //这里注意:直接修改原集合会抛出并发修改异常
            it.add("我全都要");
        }
    }
	System.out.println(list);

输出结果

[JAVA, Python, PHP, 我全都要]
第二种:不使用迭代器遍历,使用普通for遍历
    ArrayList<String> list = new ArrayList<>();
    list.add("JAVA");
    list.add("Python");
    list.add("PHP");
    for (int i = 0; i < list.size(); i++) {
        if (list.get(i).equals("PHP")) {
            list.remove("我全都要");
            list.add("C++");
        }
    }
    System.out.println(list);

输出结果

[JAVA, Python, PHP, C++]
第三种:使用toArray
    ArrayList<String> list = new ArrayList<>();
    list.add("JAVA");
    list.add("Python");
    list.add("PHP");
    Object[] obj = list.toArray();
    for (int i = 0; i < obj.length; i++) {
        if (obj[i].equals("PHP")) {
            list.add(0,"今晚学习");
        }
    }
    System.out.println(list);

输出结果

[今晚学习, JAVA, Python, PHP]

通过以上3种方法都可以解决 iterator 导致的并发修改异常。

如果非要使用迭代器修改集合,可以使用列表迭代器 ListIterator

  • 3
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值