在学习java集合的迭代器遍历集合内容,并添加内容的时候,遇到了一个ConcurrentModificationException异常,
for(Iterator<String> it=arrList.iterator();it.hasNext();){
//System.out.println(it.next()); //当for循环结束的时候,it引用就被回收器回收
Pattern p=Pattern.compile("风清扬");
Matcher m=p.matcher(it.next());
if(m.matches()){
arrList.add("白娘子");
}
}
出现了异常,查询add()方法的源码:
ensureCapacityInternal(size + 1); // Increments modCount!!
elementData[size++] = e;
return true;
这里使得modCount的加一
接着我查看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];
}
看到它调用了一个checkForComodification();方法,查看它的源码:
final void checkForComodification() {
if (modCount != expectedModCount)//重点在这!!这里因为modCount的值在add的时候,被加了1,所以不相等,所以会报错,
//这个就是为了防止在迭代没有完成的时候,对迭代器所依靠的集合进行修改!!
throw new ConcurrentModificationException();
}
这段代码的意思就是比较modCount与expectedModCount的值,若值不同则抛出之前遇到的
ConcurrentModificationException()异常!
综上所诉,当我们调用add()方法是,使得modCount的值加一,这时候与expecteModCount的值不一样了,所以报错了。再仔细想一想,这个值得作用就是预防你在
遍历集合的时候,对集合进行了增加或者删除操作,而迭代器却不知道。我们知道,迭代器是依靠集合而存在的,这明显不符合设计的初衷,所以异常的出现就不足为奇了
解决方案:
一、在你迭代到符合条件的内容是,break()方法,跳出迭代。
二、使用迭代器来进行增加或删除功能,list集合只有remove()方法可以删除内容,若要增加内容,就必须使用ArrayList特有的迭代器-ListIterator中的add()方法。
三、使用for循环,根据index来遍历数组