什么是 fail-fast?什么是 fail-safe?

💡 核心结论:一句话先记住

它们是 Java 集合在被遍历时,面对别人插队修改数据采取的两种完全相反的态度

  • fail-fast(快速失败): 脾气暴躁,眼里不容沙子。遍历时只要发现有人偷偷改了数据,立刻撂挑子不干,抛出异常
  • fail-safe(安全失败): 脾气温和,佛系包容。你想改随你改,老子复制一份副本慢慢看,绝对不抛异常。

🔍 深度大白话解析

1. 脾气暴躁的 fail-fast(代表:ArrayList、HashMap)

  • 怎么实现的: 它们内部有一个"记账本"(modCount)。只要有人增加或删除了元素,账本上的修改次数就会加 1。
  • 翻车现场: 你在用 for-each 兴高采烈地遍历元素,每走一步,都会对一下账本。如果突然发现账本数字变了(说明有人在暗中修改),它就会觉得"数据不安全了",当场甩脸子抛出经典的 ConcurrentModificationException 异常。
  • 写代码大坑: 很多人喜欢在 for 循环里直接调用 list.remove() 删元素,这就是典型的"在遍历时改账本",百分之百直接报错

2. 稳如老狗的 fail-safe(代表:CopyOnWriteArrayList、ConcurrentHashMap)

  • 怎么实现的: 它采用的是"影分身之术"。
  • 翻车现场: 当你开始遍历时,它会偷偷把当前的数据复制一份(副本)。你接下来遍历的其实是这个"副本",而别人如果去修改、删除元素,改的是"原本"。
  • 代价: 因为两者互不干扰,所以它绝对不会抛异常 。但缺点也很明显:第一,每次复制副本非常吃内存;第二,你遍历时读到的可能是"老数据"(弱一致性),别人新加进去的东西你可能看不见。

🛠️ 工作中怎么安全地"边遍历边删除"?

这是面试和写代码最常遇到的场景,有三种正确姿势:

  • 姿势一(单线程推荐): 不要用集合自带的 remove,改用迭代器自带的 iterator.remove()。因为迭代器自己删的时候会主动把账本对齐,不会触发暴脾气。
  • 姿势二(Java 8+ 推荐): 直接一行代码搞定:list.removeIf(e -> 条件),底层已经帮你封装好了安全机制。
  • 姿势三(多线程推荐): 直接换用并发安全集合,比如把 ArrayList 换成 CopyOnWriteArrayList

🎯 秒记口诀

fail-fast账本不对就抛错,直接删除必着魔;

fail-safe 复制副本不报错,数据过期要记着。

单线程删找迭代(Iterator),多线程用并发包!

相关推荐
程序员小羊!1 小时前
05 JAVA面向对象
java·开发语言
MrJson-架构师1 小时前
AgentScope Java 2.0:打造分布式、企业级智能体底座
java·开发语言·分布式
fengxin_rou1 小时前
深入理解Java类加载机制:从原理到实战详解
java·开发语言
糖果店的幽灵1 小时前
Spring AI 从入门到精通-Prompt 工程
java·spring·prompt
小江的记录本1 小时前
【Spring全家桶】Spring Cloud 2023.0.x:配置中心:Nacos Config、Apollo(附《思维导图》+《面试高频考点清单》)
java·spring boot·后端·python·spring·spring cloud·面试
weixin_408318041 小时前
2026年医疗直播行业趋势报告:技术方向、监管变化与市场格局
java·大数据·人工智能
linge_sun1 小时前
SpringAI 五步提示词大法:构建高效 AI 提示词
java·人工智能·ai编程
huipeng9261 小时前
企业级微服务开发实战(三):公共模块设计与统一规范封装
java·spring boot·spring cloud·微服务·架构·系统架构·php
我登哥MVP2 小时前
Spring Boot 从“会用”到“精通”:参数绑定体系全景
java·spring boot·spring·servlet·maven·intellij-idea·mybatis