小记一个问题,同事之前的写的代码应用跑的好好,今天突然不行了,报java.util.ConcurrentModificationException,虽然看到这个异常的时候,很快就能定位到原因,集合在循环体中直接删除了,下面用一个小例子模拟下当时的环境:
public static void main(String[] args) { ArrayList<String> lst = new ArrayList<String>(); lst.add("a"); lst.add("b"); lst.add("c"); lst.add("d"); for(String s:lst){ if(s.equals("c")){ lst.remove(s); } } }
以上代码在运行的时候不会抛任何异常。 如果改成 s.equals("a")或者其他“b”,“d” 都会报错。
下面分析其中的原因:上面的for循环,其实java在运行的时候还是调用了Iterator来循环语句的。
如下
String t = it.next();
if(t.equals("c")){
lst.remove(t);
}
}
所以看的时候根据这个更便于debug。查看源码可以看到在ArrayList的父类Abstract中的iterator()函数返回的是new Itr(),而这个Itr是内部实现了Iterator的一个私有内部类。
其中有三个重要的属性:
cursor、lastRet、execptrfModCount
其中实例化的时候
int cursor = 0; int lastRet = -1; int expectedModCount = modCount;
modCount是AbstractList中记录对象中的原始被修改的次数。
回到上面的例子。此时expectedModCount = modCount=4;当循环if为true时,即s.equals("c")的时候此时的cursor=3,在执行lst.remove("c")后,此时size=3. 下面看下hasNext的源码。
public boolean hasNext() { return cursor != size(); } public E next() { checkForComodification(); try { E next = get(cursor); lastRet = cursor++; return next; } catch (IndexOutOfBoundsException e) { checkForComodification(); throw new NoSuchElementException(); } }
此时hasNext()函数讲返回false,那么就会退出for循环。代码显然不会报错。但是如果在lst中追加了其他对象后,程序就会出现上面的异常了,出异常的点位在 next中的checkForComodification()函数。
final void checkForComodification() { if (modCount != expectedModCount) throw new ConcurrentModificationException(); } }
就是比较了modCount是否与expectedModCount一致。
但是为什么如果我们用迭代器的remove不会报错呢?
public void remove() {
if (lastRet == -1)
throw new IllegalStateException();
checkForComodification();
try {
AbstractList.this.remove(lastRet);
if (lastRet < cursor)
cursor--;
lastRet = -1;
expectedModCount = modCount;
} catch (IndexOutOfBoundsException e) {
throw new ConcurrentModificationException();
}
}
相关推荐
Java.util.ConcurrentModificationException 异常问题详解 ConcurrentModificationException 异常是 Java 中一个常见的异常,它发生在 Iterator 遍历集合时,集合同时被修改引起的异常。在 Java 中,集合类如 ...
在Java编程中,`ConcurrentModificationException`是一个常见的运行时异常,通常发生在多线程环境下对集合进行迭代和修改操作时。此问题的核心在于,Java的集合类(如ArrayList、LinkedList、HashSet等)并不支持...
ConcurrentModificationException如何解决.md
ConcurrentModificationException解决办法.md
在Java编程中,`java.util.ConcurrentModificationException` 是一个常见的运行时异常,通常发生在尝试并发修改集合时。这个异常的产生是由于集合类(如HashMap)的非线程安全特性,当你在一个线程中使用迭代器遍历...
ConcurrentModificationException(解决方案).md
在Java编程中,`ConcurrentModificationException`是一个常见的运行时异常,主要出现在多线程环境下对集合类(如List、Set、Map等)进行并发修改时。然而,这个异常不仅限于多线程环境,即使在单线程中,如果在遍历...
`java.util.ConcurrentModificationException` 是一个在 Java 中常见的运行时异常,它通常发生在多线程环境中,当一个线程正在遍历一个集合(如 `ArrayList`, `HashMap` 等),而另一个线程同时尝试修改这个集合时。...
bject[initialCapacity]; } else if (initialCapacity == 0) {...同时,需要注意在并发环境下使用ArrayList可能会遇到`ConcurrentModificationException`,应当避免在遍历过程中修改集合,或者选择线程安全的数据结构。
标题“axis1.4.1.zip”所指的是一份针对Axis1.4版本的修复补丁包,这个补丁主要是为了解决在Java Development Kit (JDK) 1.8环境下,高并发场景下出现的`ConcurrentModificationException`问题。`...
Spring数据mongodb测试 在Collections.synchronizedList或Collections.synchronizedSet上测试spring数据mongodb ConcurrentModificationException
项目中碰到的,记录一下解决方案
鸿蒙开发中碰到的报错,问题已解决,写个文档记录一下这个问题及解决方案