且构网

分享程序员开发的那些事...
且构网 - 分享程序员编程开发的那些事

13.13 java.util.ConcurrentModificationException

更新时间:2022-08-15 22:22:41

13.13 java.util.ConcurrentModificationException

问题描述

在H5性能测试平台系统的开发过程中,客户端调用服务端API,写入性能数据的时候,报了如下错误:

java.util.ConcurrentModificationException at 
java.util.ArrayList$ArrayListIterator.next(ArrayList.java:573)

错误代码:

    /**
     * 记录性能数据,写入server的数据库中
     *
     * @param requestResources
     */

    private void recordPerfData(List<HashMap<String, String>> requestResources) {
        Log.i("TAGH5", "requestResources=" + JSON.toJSONString(requestResources));
        for (HashMap<String, String> map : requestResources) {
            Log.i("TAGH5", "map=" + JSON.toJSONString(map));
            callWriteRequestResourceHttpApi(map);
        }
    }

原因分析

foreach循环,在使用iterator.hasNext()操作迭代器的时候,如果此时迭代的对象发生改变,比如插入了新数据,或者有数据被删除。此时,调用迭代器取数据ArrayListIterator.next(),会报上面的异常。

所以,涉及集合类的多线程的场景的操作的时候,要小心。

解决办法

  1. 通过Iterator修改Hashtable
while(it.hasNext()) {
Object ele = it.next();
            it.remove();
}
  1. 手动给Iterator遍历代码加锁,给修改HashMap的代码加锁。
  1. 使用CopyOnWriteArrayList

CopyOnWriteArrayList是java.util.concurrent包中的一个List的实现类。

CopyOnWrite的意思是在写时拷贝,也就是如果需要对CopyOnWriteArrayList的内容进行改变,首先会拷贝一份新的List并且在新的List上进行修改,最后将原List的引用指向新的List。

使用CopyOnWriteArrayList可以线程安全地遍历,因为如果另外一个线程在遍历的时候修改List的话,实际上会拷贝出一个新的List上修改,而不影响当前正在被遍历的List。

上面的报错代码修改如下:

recordPerfData(CopyOnWriteArrayList<HashMap<String, String>> requestResources)
  1. 如果是HashMap使用“ConcurrentHashMap”替换HashMap,ConcurrentHashMap会自己检查修改操作,对其加锁,也可针对插入操作。
import java.util.concurrent.*;

小结

遍历List的同时操作List会发生异常:

java.util.ConcurrentModificationException