在工作中碰到有个异常:java.util.ConcurrentModificationException
腾讯bugly工具给出的解决方案和说明如下:
该异常表示迭代器迭代过程中,迭代的对象发生了改变,如数据项增加或删除。
[解决方案]:由于迭代对象不是线程安全,在迭代的过程中,会检查modCount是否和初始modCount即expectedModCount一致,如果不一致,则认为数据有变化,迭代终止并抛出异常。常出现的场景是,两个线程同时对集合进行操作,线程1对集合进行遍历,而线程2对集合进行增加、删除操作,此时将会发生ConcurrentModificationException异常。
具体方法:多线程访问时要增加同步锁,或者建议使用线程安全的集合:
1. 使用ConcurrentHashMap替换HashMap,CopyOnWriteArrayList替换ArrayList;
2. 或者使用使用Vector替换ArrayList,Vector是线程安全的。Vector的缺点:大量数据操作时,由于线程安全,性能比ArrayList低. 我之前的报异常的代码为:
for (ListIterator<GridItem> it = VrAlbumGridItemDataStore.getSingleInstance().getPicList(currentPopPos).listIterator(); it.hasNext(); ) {
GridItem mGridItem = it.next();
}
注:VrAlbumGridItemDataStore.getSingleInstance().getPicList(currentPopPos)得到的List列表数据正在某个线程变化, 所有导致了该异常。 修改后的代码为:
List<GridItem> _tmpGridList = VrAlbumGridItemDataStore.getSingleInstance().getPicList(currentPopPos);
int _listSize = _tmpGridList.size();
GridItem _gridItem = null;
for (int i = 0; i < _listSize; i++) {
_gridItem = _tmpGridList.get(i);
}
修改后,该异常就没再出现了。 总结(一点经验之谈):
在以前碰到的几个报该异常的地方,解决办法都是:把数据源赋值给一个中间变量_tmpGridList,后续操作做该变量,避免直接对数据源VrAlbumGridItemDataStore.getSingleInstance().getPicList(currentPopPos)做相关操作。