在软件开发中,我们常常会遇到各种错误和异常。其中有一类比较低级但又常见的错误就是ConcurrentModificationException
异常。最近了我就写了个这种异常,这个异常通常发生在使用迭代器遍历集合时,同时对集合进行修改,从而导致迭代器检测到集合结构发生变化而抛出异常。在测试环境中可能因为数据量较小或者测试场景不充分未能显现问题,但一旦部署到生产环境,场景增多,并发操作增多,这个低级错误就会爆发。
在使用entrySet()遍历Map时,返回的是Map的EntrySet视图,它与原始的Map是关联的。在迭代的过程中,如果我们直接通过params.remove(entry.getKey())去修改Map,会导致EntrySet视图与原始Map的结构不一致,从而抛出ConcurrentModificationException(并发修改异常)。
for (Map.Entry<String, Set<String>> entry : params.entrySet()) {
if(KEY_SET.contains(entry.getKey())){
executor.execute(()-> this.doFlush(entry.getKey(),entry.getValue()) );
params.remove(entry.getKey());
}
}
Caused by: java.util.ConcurrentModificationException: null
at java.util.LinkedHashMap$LinkedHashIterator.nextNode(LinkedHashMap.java:719)
at java.util.LinkedHashMap$LinkedEntryIterator.next(LinkedHashMap.java:752)
at java.util.LinkedHashMap$LinkedEntryIterator.next(LinkedHashMap.java:750)
at cn.xj.common.utils.http.cache.CacheConsumer.flushCache(CacheConsumer.java:100)
at cn.xj.common.utils.http.cache.CacheConsumer.msgConsumer(CacheConsumer.java:83)
at cn.xj.framework.task.JobTask.cacheFlushJob(JobTask.java:1829)
... 10 common frames omitted
解决这个问题的方法是,使用迭代器来进行安全的删除操作。具体代码如下:
Iterator<Map.Entry<String, Set<String>>> iterator = params.entrySet().iterator();
while (iterator.hasNext()) {
Map.Entry<String, Set<String>> entry = iterator.next();
if (KEY_SET.contains(entry.getKey())) {
executor.execute(() -> this.doFlush(entry.getKey(), entry.getValue()));
iterator.remove(); // 使用迭代器的remove方法来安全地删除元素
}
}
这样就能避免ConcurrentModificationException异常。
作为开发者,避免低级错误同样重要。希望本文能帮助读者更好地理解并解决ConcurrentModificationException
异常,以及在开发中提高代码质量,减少不必要的问题发生。同时,重视测试工作,让我们的项目在实际应用中更加稳定和可靠。