美文网首页
List集合增强for循环时修改集合集合长度时产生的异常

List集合增强for循环时修改集合集合长度时产生的异常

作者: flyjar | 来源:发表于2022-07-26 13:26 被阅读0次

1.介绍

在List 集合使用增强for循环遍历时,我们如果改变了集合的长度,会抛出异常。下面举个例子:

    public static void main(String[] args){
        List<String> list = new ArrayList<>();
        list.add("a");
        list.add("b");
        list.add("c");

        for (String s : list) {
            if("a".equals(s)){
                //list.remove(s);
                list.add("a++");
            }
        }
    }

异常信息:

Exception in thread "main" java.util.ConcurrentModificationException
    at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:901)
    at java.util.ArrayList$Itr.next(ArrayList.java:851)
    ···

2.增强for循环的原理

增强for循环是Java提供的语法糖,比如

for (Integer i : list) {
   System.out.println(i);
}

反编译后,其实是这样的:

Integer i;
for(Iterator iterator = list.iterator(); iterator.hasNext(); System.out.println(i)){
   i = (Integer)iterator.next();        
}

3.现在我们通过源代码来分析一开始例子中所报的错误:

位置是在ArrayList的850行和899行
iterator的next方法


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];
}

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

抛出异常的条件是 modCount != expectedModCount 。

提到的几个关键变量:

  • cursor:表示下一个要访问的元素的索引,从next()方法的具体实现就可看出
  • lastRet:表示上一个访问的元素的索引
  • expectedModCount:表示对ArrayList修改次数的期望值,它的初始值为modCount。
  • modCount是AbstractList类中的一个成员变量

modCount表示对List的修改次数,他的初始值是0。查看ArrayList的add()和remove()方法就可以发现,每次调用add()方法或者remove()方法就会对modCount进行加1操作。


public boolean add(E e) {
    ensureCapacityInternal(size + 1);  // Increments modCount!!
    elementData[size++] = e;
    return true;
}

private void ensureCapacityInternal(int minCapacity) {
    if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
        minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);
    }

    ensureExplicitCapacity(minCapacity);
}

private void ensureExplicitCapacity(int minCapacity) {
    modCount++;

    // overflow-conscious code
    if (minCapacity - elementData.length > 0)
        grow(minCapacity);
}

看到这里,问题的原因找到了,在直接调用add()或者remove()方法时,只修改了modCount,却没有修改expectedModCount!

ArrayList中给出的解决办法是提供了一个ListIterator:


public ListIterator<E> listIterator() {
    return new ListItr(0);
}

private class ListItr extends Itr implements ListIterator<E> {}

它实现的迭代器实现了add(E e) 和remove()方法,可以解决抛出异常的问题,这两个方法里都修改了expectedModCount,让他和modCount保持一致。

重点:所以在遍历list去操作list的时候,调用迭代器的remove或者add方法进行对list长度进行修改,而不是直接调用list的remove或者add方法

相关文章

  • List集合增强for循环时修改集合集合长度时产生的异常

    1.介绍 在List 集合使用增强for循环遍历时,我们如果改变了集合的长度,会抛出异常。下面举个例子: 异常信息...

  • List集合

    List集合特有的方法 list集合可以用迭代器和for循环进行遍历 迭代器并发修改异常 ListIterator...

  • 集合常用方法

    1-获取集合长度-list.length 2-获取集合大小-list.size 3-循环遍历-foreach 4-...

  • 【Java】【集合框架】集合框架(list)

    集合框架(list) ArrayList(数组集合)查询与修改快 迭代器并发修改异常解决方案 LinkedList...

  • Java集合的用法及特点

    单列LIst集合 单列列表集合: 集合的由来长度是固定的,当添加的元素超过了多个的长度时需要对分段重新定义,太麻烦...

  • list集合遍历方式

    Java中List集合的三种遍历方式:①for循环遍历:指定下标长度,使用List集合的size()方法,进行fo...

  • 16.Collection集合

    主要内容: Collection 集合 迭代器 增强for List 集合 Set 集合 1,集合 集合是java...

  • 04.并发修改异常

    并发修改异常 当使用迭代器遍历集合的时候,使用了集合中的 增加/删除 方法,导致并发修改异常产生 并发修改异常解决...

  • 寒假12:List

    List集合的概述和特点 List特有方法 并发修改异常 ListIterator

  • java for循环是否需要判空

    集合为null时会报空指针异常;长度为0的空集合不会;普通for循环原理一样

网友评论

      本文标题:List集合增强for循环时修改集合集合长度时产生的异常

      本文链接:https://www.haomeiwen.com/subject/uisoirtx.html