当前位置: 首页 > news >正文

Java 集合遍历过程中修改数据触发 Fail-Fast 机制 ,导致报ConcurrentModificationException异常

Java Fail-Fast 机制

Fail-Fast 机制是 Java 集合框架中的一种错误检测机制,用于在遍历集合时检测结构修改。如果在迭代器创建之后,集合被修改(例如添加或删除元素),并且这种修改不是通过迭代器自身的 remove() 方法进行的,那么迭代器会立即抛出 ConcurrentModificationException 异常,以防止不一致或不可预测的行为。

工作原理
  1. 修改计数器

    • 集合类(如 ArrayListHashMap 等)内部维护一个 modCount 计数器,记录集合被结构性修改的次数(结构性修改包括添加或删除元素,但不包括通过迭代器自身的 remove() 方法进行的删除)。
  2. 迭代器的预期修改计数

    • 当创建迭代器时,迭代器会记录当前集合的 modCount 值,作为其 expectedModCount
    • 在每次调用迭代器的 next() 方法时,迭代器会检查 expectedModCount 是否与集合的当前 modCount 一致。
  3. 检测不一致

    • 如果在迭代过程中,集合的 modCount 发生变化(即 expectedModCount 不等于 modCount),迭代器会立即抛出 ConcurrentModificationException 异常。
示例代码
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;

public class FailFastExample {
    public static void main(String[] args) {
        List<String> list = new ArrayList<>();
        list.add("A");
        list.add("B");
        list.add("C");

        Iterator<String> iterator = list.iterator();
        while (iterator.hasNext()) {
            String element = iterator.next();
            System.out.println(element);
            // 在迭代过程中修改集合,会抛出 ConcurrentModificationException
            list.add("D");
        }
    }
}

输出

A
Exception in thread "main" java.util.ConcurrentModificationException
    at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:909)
    at java.util.ArrayList$Itr.next(ArrayList.java:859)
    at FailFastExample.main(FailFastExample.java:13)
解决方法
  1. 使用迭代器自身的 remove() 方法

    • 如果需要在遍历过程中删除元素,应使用迭代器的 remove() 方法,而不是直接操作集合。
    Iterator<String> iterator = list.iterator();
    while (iterator.hasNext()) {
        String element = iterator.next();
        if (element.equals("B")) {
            iterator.remove(); // 安全删除元素
        }
    }
    
  2. 使用线程安全的集合类

    • 使用 java.util.concurrent 包中的线程安全集合类,如 CopyOnWriteArrayListConcurrentHashMap 等。
    import java.util.concurrent.CopyOnWriteArrayList;
    
    public class ConcurrentExample {
        public static void main(String[] args) {
            CopyOnWriteArrayList<String> list = new CopyOnWriteArrayList<>();
            list.add("A");
            list.add("B");
            list.add("C");
    
            Iterator<String> iterator = list.iterator();
            while (iterator.hasNext()) {
                String element = iterator.next();
                System.out.println(element);
                // 在迭代过程中修改集合,不会抛出 ConcurrentModificationException
                list.add("D");
            }
        }
    }
    
  3. 使用 Collections.synchronizedList()Collections.synchronizedSet()

    • 将集合包装为线程安全的集合。
    import java.util.Collections;
    import java.util.Iterator;
    import java.util.List;
    
    public class SynchronizedExample {
        public static void main(String[] args) {
            List<String> list = Collections.synchronizedList(new ArrayList<>());
            list.add("A");
            list.add("B");
            list.add("C");
    
            synchronized (list) {
                Iterator<String> iterator = list.iterator();
                while (iterator.hasNext()) {
                    String element = iterator.next();
                    System.out.println(element);
                    // 在迭代过程中修改集合,不会抛出 ConcurrentModificationException
                    list.add("D");
                }
            }
        }
    }
    
注意事项
  1. 单线程环境

    • 在单线程环境中,Fail-Fast 机制有助于及时发现集合被意外修改的问题。
    • 但需要注意在迭代过程中不要直接修改集合,除非使用迭代器自身的 remove() 方法。
  2. 多线程环境

    • Fail-Fast 机制在多线程环境中可能会导致 ConcurrentModificationException 异常。
    • 应使用线程安全的集合类或同步机制来避免此类问题。
  3. 性能影响

    • Fail-Fast 机制本身对性能的影响较小,主要体现在每次迭代时的 modCount 检查。
    • 但在多线程环境下,频繁的同步操作可能会显著影响性能。
总结
  • Fail-Fast 机制 是 Java 集合框架中用于检测集合在迭代过程中被修改的一种机制。
  • 通过在迭代过程中抛出 ConcurrentModificationException 异常,Fail-Fast 机制可以及时发现不一致的行为,确保集合的完整性和一致性。
  • 在使用 Fail-Fast 机制时,需要注意在迭代过程中不要直接修改集合,除非使用迭代器自身的 remove() 方法。
  • 对于多线程环境,建议使用线程安全的集合类或同步机制来避免 ConcurrentModificationException 异常。

通过合理使用 Fail-Fast 机制,可以提高代码的健壮性和可靠性。

相关文章:

  • 电脑实用小工具推荐--屏幕录制软件Bandicam(班迪录屏)
  • ECharts中Map(地图)样式配置、渐变色生成
  • C语言交换两数
  • Dijkstra算法
  • 【蓝桥】模拟
  • Day16:字符串的排列
  • eBPF 实时捕获键盘输入
  • Day2 导论 之 「存储器,IO,微机工作原理」
  • 【测试篇】打破测试认知壁垒,从基础概念起步
  • 零基础上手Python数据分析 (5):Python文件操作 - 轻松读写,数据导入导出不再是难题
  • 【SpringMVC】常用注解:@RequestHeader
  • sentinel限流算法
  • 《DeepSeek深度使用教程:开启智能交互新体验》Deepseek深度使用教程
  • 第五章 树、2叉树
  • 這是我第一次寫關於aapenal服務器管理控制面板的文章
  • “个人陈述“的“十要“和“十不要“
  • 1、操作系统引论
  • Certbot实现SSL免费证书自动续签(CentOS 7 + nginx/apache)
  • (undone) 梳理 xv6-lab-2023 fs.img 生成过程,以及文件系统结构
  • QT编程之QStackedWidget
  • 案件发回重审,李在明参选韩总统之路再添波折
  • 4月一二线城市新房价格环比上涨,沪杭涨幅居百城前列
  • 强制性国家标准《危险化学品企业安全生产标准化通用规范》发布
  • 中国海警局新闻发言人就菲律宾非法登临铁线礁发表谈话
  • 党旗下的青春|83岁仍在“下生活”,他说生活是创作的源泉
  • 高璞任中国一汽党委常委、副总经理