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

快速失败(fail-fast)和安全失败(fail-safe)的区别

        在 Java 中,‌快速失败(Fail-Fast)‌和‌安全失败(Fail-Safe)‌是集合类(Collection)在迭代过程中处理并发修改的两种不同策略,二者的核心区别在于 ‌对并发修改的感知机制与容错性‌。


‌一、 快速失败(Fail-Fast)

  • 核心特性‌:
    一旦检测到集合在迭代过程中被修改(除通过迭代器自身的方法),立即抛出 ConcurrentModificationException,终止程序。

  • 实现原理‌:

    • 迭代器内部维护一个 modCount 计数器,记录集合的修改次数。
    • 每次迭代时检查 modCount 是否与初始值一致,若不一致,说明集合被修改,触发异常。
  • 适用场景‌:
    单线程环境,或明确确保不会在迭代时修改集合的场景。

  • 示例集合‌:
    ArrayListHashMapHashSet(非线程安全集合的默认行为)。

  • 代码演示‌:

List<String> list = new ArrayList<>(Arrays.asList("A", "B", "C"));
Iterator<String> it = list.iterator();
while (it.hasNext()) {String s = it.next();list.remove(s); // 直接操作原集合:抛出 ConcurrentModificationException
}

‌二、安全失败(Fail-Safe)

  • 核心特性‌:
    允许在迭代过程中修改集合(如添加、删除元素),不会抛出异常。迭代器基于集合的‌副本‌或‌快照‌进行遍历,与原集合解耦。

  • 实现原理‌:

    • 迭代器操作的是集合的副本(如 CopyOnWriteArrayList 的写时复制技术)。
    • 或通过并发控制(如 ConcurrentHashMap 的分段锁机制)保证线程安全。
  • 适用场景‌:
    多线程环境,或需要在迭代过程中修改集合的场景。

  • 示例集合‌:
    CopyOnWriteArrayListConcurrentHashMapConcurrentLinkedQueue

  • 代码演示‌:

List<String> list = new CopyOnWriteArrayList<>(Arrays.asList("A", "B", "C"));
Iterator<String> it = list.iterator();
while (it.hasNext()) {String s = it.next();list.add("D"); // 直接操作原集合:不会抛出异常
}
System.out.println(list); // 输出 [A, B, C, D, D, D]

三、关键区别总结

特性快速失败(Fail-Fast)安全失败(Fail-Safe)
并发修改检测立即抛出异常,强制终止程序允许修改,不触发异常
迭代器数据源直接操作原集合基于集合的副本或快照
性能开销低(仅计数器检查)高(需维护副本或同步控制)
数据一致性强一致性(反映最新状态)弱一致性(可能遍历旧数据)
典型集合ArrayListHashMapCopyOnWriteArrayListConcurrentHashMap

四、选择建议

  • 若需 ‌强一致性‌ 且确定无并发修改 → ‌Fail-Fast‌。
  • 若需 ‌高并发支持‌ 或允许弱一致性 → ‌Fail-Safe‌。

  例如:

  • 单线程场景优先使用 ArrayList(Fail-Fast);
  • 多线程高并发场景选择 ConcurrentHashMap(Fail-Safe)。

五、Fail-Safe的使用场景

        Fail-Safe(安全失败)机制在多线程或动态修改集合的场景中尤为重要,以下是典型使用场景及示例:

(1)‌多线程并发操作

        当多个线程同时读写集合时,Fail-Safe集合(如 ConcurrentHashMap)可避免因并发修改导致的异常。
示例:

ConcurrentHashMap<String, Integer> map = new ConcurrentHashMap<>();
map.put("A", 1);
map.put("B", 2);// 线程1:迭代
new Thread(() -> {Iterator<String> it = map.keySet().iterator();while (it.hasNext()) {System.out.println(it.next()); // 安全遍历}
}).start();// 线程2:修改
new Thread(() -> {map.put("C", 3); // 不会触发异常
}).start();
(2)动态数据更新

        在需要实时更新数据的场景(如实时监控系统),CopyOnWriteArrayList 允许遍历时修改数据。

示例‌:

CopyOnWriteArrayList<String> logList = new CopyOnWriteArrayList<>();
logList.add("Log1");// 遍历时追加日志
for (String log : logList) {System.out.println(log);logList.add("NewLog"); // 安全操作
}
‌(3)高吞吐量系统

        如消息队列消费者处理数据时,Fail-Safe集合可避免因频繁修改导致的迭代中断。

相关文章:

  • 传统医疗系统文档集中标准化存储和AI智能化更新路径分析
  • 爬虫知识之IP代理
  • 用 AI 让学习更懂你:如何打造自动化个性化学习系统?
  • 嵌入式开发方向开发利器
  • OpenLayers 加载全屏显示控件
  • 【萤火工场GD32VW553-IOT开发板】ADC电压的LabVIEW采集
  • 【Code Agent Benchmark】论文分享No.15:TAU-Bench
  • 标准版v5.6.1, 优化了一些细节提升体验
  • RabbitMQ的详细使用
  • 超详细网络介绍(超全)
  • 5.24本日总结
  • CQF预备知识:Python相关库 -- NumPy 基础知识 - 数组创建
  • Ubuntu20.04 gr-gsm完整安装教程
  • SQL每日一题
  • SQL SERVER常用聚合函数整理及示例
  • 全面指南:使用Node.js和Python连接与操作MongoDB
  • 二十五、面向对象底层逻辑-SpringMVC九大组件之HandlerMapping接口设计
  • 【Linux】进程基本概念与基本操作
  • 【深度学习新浪潮】如何用Dify构建自己的AI Agent?
  • 利用 XML 外部实体注入(XXE)读取文件和探测内部网络
  • 北京模板网站制作/关键词怎样做优化排名
  • 追星做网站/zoho crm
  • 自家电脑做网站服务器w7花生壳/地方网站建设
  • 记事本做网站怎么改字体颜色/百度seo 优化
  • 网站强制字体wordpress/网站建设公司苏州
  • 一个虚拟空间做两个网站/武汉seo论坛