网站首页 > 技术文章 正文
原文链接:
https://mp.weixin.qq.com/s/dPA7zohM2LfBrZT-sfQnyg
你真的了解 fail-fast和 fail-safe吗?
简介
java.util 包下的 属于 fail-fast , 快速失败~
java.util.concurrent 包下的 属于 fail-safe ,安全失败~
简单来说 就是 fail-fast 在迭代时,如果发现 该集合数据 结构被改变 (modCount != expectedModCount),就会 抛出
ConcurrentModificationException
小伙伴们可以参考下 下面的代码简单实验一下~
fail-fast实验代码
实验对象是 Hashtable,这里采用 jdk1.7 的写法 ~
因为博主还在研究 下文中 ConcurrentHashMap 在7和8中有啥不一样
class E implements Runnable{
Hashtable<String, String> hashtable;
public E(Hashtable<String, String> hashtable) {
this.hashtable = hashtable;
}
private void add(Hashtable<String, String> hashtable){
for (int i = 0; i < 10000000; i++) {
hashtable.put("a",""+i);
}
}
@Override
public void run() {
add(hashtable);
}
}
public class D {
public static void main(String[] args) {
Hashtable<String, String> hashtable = new Hashtable<String, String>();
hashtable.put("1","2");
hashtable.put("2","2");
hashtable.put("3","2");
hashtable.put("4","2");
hashtable.put("15","2");
new Thread(new E(hashtable)).start();
Set<Map.Entry<String, String>> entries = hashtable.entrySet();
Iterator<Map.Entry<String, String>> iterator = entries.iterator();
try {
Thread.sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
while (iterator.hasNext()){
System.out.println(iterator.next());
iterator.remove();
}
}
}
效果如图:
触发的原理:
当集合数据结构发生变化时,这两个值是不相等的,所以会抛出该异常~ 。
结论:
虽然 HashTable 是 线程安全的 , 但是它有 fail-fast 机制 ,所以在多线程情况下进行 迭代 也不能去修改它的数据结构!
fail-fast 机制 不允许并发修改!
fail-safe实验代码
class E implements Runnable{
ConcurrentHashMap<String, String> concurrentHashMap;
public E(ConcurrentHashMap<String, String> concurrentHashMap) {
this.concurrentHashMap = concurrentHashMap;
}
private void add( ConcurrentHashMap<String, String> concurrentHashMap){
for (int i = 0; i < 100000; i++) {
concurrentHashMap.put("a"+i,""+i);
}
}
@Override
public void run() {
add(concurrentHashMap);
}
}
public class D {
public static void main(String[] args) {
ConcurrentHashMap<String, String> concurrentHashMap = new ConcurrentHashMap<String, String>();
concurrentHashMap.put("1","2");
concurrentHashMap.put("2","2");
concurrentHashMap.put("3","2");
concurrentHashMap.put("4","2");
concurrentHashMap.put("15","2");
new Thread(new E(concurrentHashMap)).start();
try {
Thread.sleep(2);
} catch (InterruptedException e) {
e.printStackTrace();
}
Set<Map.Entry<String, String>> entries = concurrentHashMap.entrySet();
for (Map.Entry<String, String> entry : entries) {
System.out.println(entry);
// 这里不用调用 iterator 去 remove
concurrentHashMap.remove(entry.getKey());
}
}
}
效果如图:
代码运行讲解,线程A 往里加数据,线程B 遍历它的数据,并删除。
可以看到这里并没有报错~,但是它也不能保证遍历到所有的值 (可以理解为无法获取到最新的值)
有没有感受到一丝丝 安全失败的感觉~
哈哈哈 它的特点就是 允许并发修改,不会抛出
ConcurrentModificationException ,但是无法保证拿到的是最新的值
不知道小伙伴们看完上面的实验代码有没有疑惑
(((*)
为什么可以调用它自身的 remove呢?
别急~ 我们先来看看使用这个迭代器中发生了什么?
源码走起~
小伙伴们可以看看下面四张图~
创建迭代器的过程
从 图一 可以看到会去创造一个 EntryIterator, 而 它又 继承了 HashIterator ,在初始化时,会先调用父类的构造器。
从 图三 可以发现 HashIterator 在初始化 时,会去调用 advance 方法 (这里就不展开这个 concurrentHashMap结构啦~ ) 这里的重点在最后一张图 , 它调用的是 UNSAFE.getObjectVolatile 。
它的作用是 强制从主存中获取属性值。
小伙伴们可以自行对比下 HashMap 或者 上面的 HashTable,他们都是直接 拿到代码中定义的这个 Entry[]~。
不知道小伙伴们 get 得到这个点没有~
哈哈哈 容我唠叨唠叨一下~
4ye 在网上搜这个 fail-fast 和 fail-safe 的区别时,看到下面这张图。
几乎都在说 fail-safe 会复制原来的集合,然后在复制出来的集合上进行操作,然后就说这样是不会抛出
ConcurrentModificationException 异常了。
可是这种说法是 不严谨的~ 它描述的情况应该是针对这个 CopyOnWriteArrayList 或者 CopyOnWriteArraySet 的情况(下面的源码讲到~)
CopyOnWriteArrayList 源码
可以发现这里 snapshot 的指针是始终指向这个原数组的(当你创建迭代器的时候)
当你添加数据时,它会复制原来的数组,并在复制出来的数组上进行修改,
然后再设置进去,可以发现至始至终都没有修改到这个原数组,
所以迭代器中的数据是不受影响的~
结论
fail-safe 也是得具体情况具体分析的。
- 如果是 CopyOnWriteArrayList 或者 CopyOnWriteArraySet ,就属于 复制原来的集合,然后在复制出来的集合上进行操作 的情况 ,所以是不会抛出这个 ConcurrentModificationException 的 。
- 如果是这个 concurrentHashMap 的,就比较硬核了~ 它直接操作底层,调用UNSAFE.getObjectVolatile ,直接 强制从主存中获取属性值,也是不会抛出这个 ConcurrentModificationException 的 。
- 并发下,无法保证 遍历时拿到的是最新的值~
嘿嘿 现在回答上面那个 为啥可以 remove 的问题啦~
remove的源码如下
重点在红框处, pred 为 null 表示是数组的头部,此时调用 setEntryAt ,这里也是出现了这个 UNSAFE
UNSAFE.putOrderedObject 这段代码的意思就是 :
有序的(有延迟的) 强制 更新数据到 主内存。(不能立刻被其他线程发现)
这些 和 Java 的 JMM (Java内存模型)有关!
总结
java.util 包下的属于fail-fast ,
特点:
不允许并发修改,如果并发修改的话会导致在迭代过程中抛出
ConcurrentModificationException ,
出发点是 modCount != expectedModCount 。
java.util.concurrent 包下的 属于 fail-safe ,
特点:
允许并发修改,但是 无法保证在迭代过程中获取到最新的值 。
concurrentHashMap 获取和修改数据时 ,是通过 UNSAFE 类 直接从主内存中获取或者更新数据到主内存~
CopyOnWriteArrayList 或者 CopyOnWriteArraySet ,就直接 复制原来的集合,然后在复制出来的集合上进行操作
。
- 上一篇: 办公小技巧:Word“安全模式”用通透
- 下一篇:已经是最后一篇了
猜你喜欢
- 2025-08-31 办公小技巧:Word“安全模式”用通透
- 2025-08-31 新来的妹子误执行 “rm -rf” !_七零大院新来的小美人是黑道千金
- 2025-05-28 iOS 14.7正式版发布:为什么叫FCS?为什么没有iPadOS?
- 2025-05-28 排忧解难 解决Windows 10安全模式三大问题
- 2025-05-28 1分钟速看!苹果新品发布会看点一览
- 2025-05-28 蔡健雅公司发声明否认新歌抄袭,要求停止造谣
- 2025-05-28 数据恢复工具
- 2025-05-28 车载科技警报防止孩子遗留车内
- 2025-05-28 设计模式:单例模式及C及C++实现示例
- 2025-05-28 澳政府斥巨资开发新冠追踪APP:仅发现17例 还被曝收集隐私
- 最近发表
-
- fail-safe 和 fail-fast 都是什么鬼?
- 办公小技巧:Word“安全模式”用通透
- 新来的妹子误执行 “rm -rf” !_七零大院新来的小美人是黑道千金
- 如何利用 Python 自动发邮件,打工人福音
- Python内置模块base64 :Base16, Base32, Base64, Base85 编码详解
- java调用API操作GitLab_java调用git的接口
- spingboot 实现导入excel数据生成二维码
- SpringBoot中7个文件上传下载工具
- java项目中接入大模型,简历必备_java介绍项目中做过的模块
- spring boot-MultipartFile 机制_spring boot multipartfile为null
- 标签列表
-
- cmd/c (90)
- c++中::是什么意思 (84)
- 标签用于 (71)
- 主键只能有一个吗 (77)
- c#console.writeline不显示 (95)
- pythoncase语句 (88)
- es6includes (74)
- sqlset (76)
- windowsscripthost (69)
- apt-getinstall-y (100)
- node_modules怎么生成 (87)
- chromepost (71)
- flexdirection (73)
- c++int转char (80)
- mysqlany_value (79)
- static函数和普通函数 (84)
- el-date-picker开始日期早于结束日期 (76)
- asynccallback (71)
- localstorage.removeitem (74)
- vector线程安全吗 (70)
- java (73)
- js数组插入 (83)
- mac安装java (72)
- 查看mysql是否启动 (70)
- 无效的列索引 (74)