避免低级错误:深入解析Java的ConcurrentModificationException异常

简介: 在软件开发中,我们常常会遇到各种错误和异常。其中有一类比较低级但又常见的错误就是`ConcurrentModificationException`异常。最近了我就写了个这种异常,这个异常通常发生在使用迭代器遍历集合时,同时对集合进行修改,从而导致迭代器检测到集合结构发生变化而抛出异常。在测试环境中可能因为数据量较小或者测试场景不充分未能显现问题,但一旦部署到生产环境,场景增多,并发操作增多,这个低级错误就会爆发。

concurrent.jpg

在软件开发中,我们常常会遇到各种错误和异常。其中有一类比较低级但又常见的错误就是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异常。

避免类似问题的方法

  • 使用迭代器:在遍历集合时,如果需要对集合进行修改操作,请使用迭代器的remove()方法来进行安全的删除操作。

  • 使用CopyOnWrite容器:如果可能,在多线程环境下,可以考虑使用Java提供的线程安全容器,比如CopyOnWriteArrayList或ConcurrentHashMap,它们内部实现了并发安全,可以避免ConcurrentModificationException。

  • 合理规划数据操作:在处理数据时,尽量避免在遍历过程中进行删除操作,可以先标记要删除的元素,然后在遍历结束后,再进行删除操作。

  • 使用同步块:在多线程环境下,如果无法使用线程安全容器,可以使用同步块(synchronized)来保护对集合的修改操作,确保在修改时不会被其他线程干扰。

  • 测试覆盖:在测试环境中尽量模拟真实的生产环境数据,测试各种可能的情况,以确保代码在生产环境能够正常运行。

总结:

作为开发者,避免低级错误同样重要。希望本文能帮助读者更好地理解并解决ConcurrentModificationException异常,以及在开发中提高代码质量,减少不必要的问题发生。同时,重视测试工作,让我们的项目在实际应用中更加稳定和可靠。

目录
相关文章
|
5月前
|
机器学习/深度学习 JSON Java
Java调用Python的5种实用方案:从简单到进阶的全场景解析
在机器学习与大数据融合背景下,Java与Python协同开发成为企业常见需求。本文通过真实案例解析5种主流调用方案,涵盖脚本调用到微服务架构,助力开发者根据业务场景选择最优方案,提升开发效率与系统性能。
1271 0
|
5月前
|
Java
Java的CAS机制深度解析
CAS(Compare-And-Swap)是并发编程中的原子操作,用于实现多线程环境下的无锁数据同步。它通过比较内存值与预期值,决定是否更新值,从而避免锁的使用。CAS广泛应用于Java的原子类和并发包中,如AtomicInteger和ConcurrentHashMap,提升了并发性能。尽管CAS具有高性能、无死锁等优点,但也存在ABA问题、循环开销大及仅支持单变量原子操作等缺点。合理使用CAS,结合实际场景选择同步机制,能有效提升程序性能。
|
5月前
|
Java 开发者
Java并发编程:CountDownLatch实战解析
Java并发编程:CountDownLatch实战解析
504 100
|
4月前
|
存储 安全 Java
《数据之美》:Java集合框架全景解析
Java集合框架是数据管理的核心工具,涵盖List、Set、Map等体系,提供丰富接口与实现类,支持高效的数据操作与算法处理。
|
5月前
|
Java 开发者
Java 函数式编程全解析:静态方法引用、实例方法引用、特定类型方法引用与构造器引用实战教程
本文介绍Java 8函数式编程中的四种方法引用:静态、实例、特定类型及构造器引用,通过简洁示例演示其用法,帮助开发者提升代码可读性与简洁性。
|
4月前
|
存储 人工智能 算法
从零掌握贪心算法Java版:LeetCode 10题实战解析(上)
在算法世界里,有一种思想如同生活中的"见好就收"——每次做出当前看来最优的选择,寄希望于通过局部最优达成全局最优。这种思想就是贪心算法,它以其简洁高效的特点,成为解决最优问题的利器。今天我们就来系统学习贪心算法的核心思想,并通过10道LeetCode经典题目实战演练,带你掌握这种"步步为营"的解题思维。
|
5月前
|
安全 Java API
Java SE 与 Java EE 区别解析及应用场景对比
在Java编程世界中,Java SE(Java Standard Edition)和Java EE(Java Enterprise Edition)是两个重要的平台版本,它们各自有着独特的定位和应用场景。理解它们之间的差异,对于开发者选择合适的技术栈进行项目开发至关重要。
707 1
|
11月前
|
算法 测试技术 C语言
深入理解HTTP/2:nghttp2库源码解析及客户端实现示例
通过解析nghttp2库的源码和实现一个简单的HTTP/2客户端示例,本文详细介绍了HTTP/2的关键特性和nghttp2的核心实现。了解这些内容可以帮助开发者更好地理解HTTP/2协议,提高Web应用的性能和用户体验。对于实际开发中的应用,可以根据需要进一步优化和扩展代码,以满足具体需求。
1061 29
|
11月前
|
前端开发 数据安全/隐私保护 CDN
二次元聚合短视频解析去水印系统源码
二次元聚合短视频解析去水印系统源码
451 4
|
11月前
|
JavaScript 算法 前端开发
JS数组操作方法全景图,全网最全构建完整知识网络!js数组操作方法全集(实现筛选转换、随机排序洗牌算法、复杂数据处理统计等情景详解,附大量源码和易错点解析)
这些方法提供了对数组的全面操作,包括搜索、遍历、转换和聚合等。通过分为原地操作方法、非原地操作方法和其他方法便于您理解和记忆,并熟悉他们各自的使用方法与使用范围。详细的案例与进阶使用,方便您理解数组操作的底层原理。链式调用的几个案例,让您玩转数组操作。 只有锻炼思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~

推荐镜像

更多
  • DNS