Java集合如何保证线程安全?Fail-fast、Fail-safe机制是什么?

有个学弟在面快手的时候,被问到Java非线程安全的集合,如HashMap,ArrayList是如何保证线程安全的时候犯了难,也没想到会问这么细,这篇文章将很快的说清楚这里的逻辑,发车~

一些源码

首先我们看看HashMap的一些关键代码: 我们注意到会有一个ModCount (修改计数)的属性,简而言之,容器在修改的时候会检查这个modCount是否与自己当前应该的modCount是否相同,不同的话就会快速失败,抛出ConcurrentModificationException,详情看下面的代码:

java 复制代码
public final void remove() {  
    Node<K,V> p = current;  
    if (p == null)  
        throw new IllegalStateException();  
    if (modCount != expectedModCount)  
        throw new ConcurrentModificationException();  
    current = null;  
    K key = p.key;  
    removeNode(hash(key), key, null, false, false);  
    expectedModCount = modCount;  
}

看这个方法就很清楚了,其他非线程安全集合如ArrayList也是这么处理的,ArrayList中直接封装了一个判断的方法:

java 复制代码
final void checkForComodification() {  
    if (modCount != expectedModCount)  
        throw new ConcurrentModificationException();  
}

Fail-fast

望文生义,fail-fast快速失败机制,是多线程并发操作集合时的一种失败处理机制。

在集合遍历过程中,一旦发现容器中的数据被修改了,会立刻抛出ConcurrentModificationException异常,从而导致遍历失败。

这里如果我一边遍历集合,一边添加元素,是会报ConcurrentModificationException的。

Fail-safe

Fail-safe:表示失败安全,也就是在这种机制下,出现集合元素的修改,不会抛出ConcurrentModificationException

原因是采用安全失败机制的集合容器,在遍历时不是直接在集合内容上访问的,而是先复制原有集合内容,

在拷贝的集合上进行遍历。由于迭代时是对原集合的拷贝进行遍历,所以在遍历过程中对原集合所作的修改并不能被迭代器检测到

比如这种情况

java 复制代码
public class Test {  
    public static void main(String[] args) {  
        CopyOnWriteArrayList<Integer> list = new CopyOnWriteArrayList<>(new Integer[]{1,7,9,11});  
        Iterator itr = list.iterator();  
        while(itr.hasNext()) {  
            Integer i = (Integer) itr.next();  
            System.out.println(i);  //是不会打印15的
            if(i==7){  
                list.add(15);  
            }  
        }  
    }  
}

定义了一个CopyOnWriteArrayList,在对这个集合遍历过程中,对集合元素做修改后,不会抛出异常,但同时也不会打印出增加的元素。

java.util.concurrent包下的容器都是安全失败的,可以在多线程下并发使用,并发修改。

常见的的使用Fail-safe方式遍历的容器有ConcerrentHashMap和CopyOnWriteArrayList等。

相关推荐
爱笑的眼睛1110 分钟前
深入解析Matplotlib Axes API:构建复杂可视化架构的核心
java·人工智能·python·ai
LYFlied22 分钟前
【算法解题模板】动态规划:从暴力递归到优雅状态转移的进阶之路
数据结构·算法·leetcode·面试·动态规划
2501_9216494923 分钟前
免费获取股票历史行情与分时K线数据 API
开发语言·后端·python·金融·数据分析
乐观甜甜圈24 分钟前
JDK8 中线程实现方法与底层逻辑详解
java
尤物程序猿28 分钟前
Java如何不建表完成各种复杂的映射关系(鉴权概念、区域概念、通用概念)
java·开发语言
cike_y40 分钟前
JSP内置对象及作用域&双亲委派机制
java·前端·网络安全·jsp·安全开发
也许是_43 分钟前
大模型应用技术之 Spring AI 2.0 变更说明
java·人工智能·spring
xunyan62341 小时前
面向对象(下)-内部类的分类
java·学习
巴拉巴拉~~1 小时前
KMP 算法通用进度条组件:KmpProgressWidget 多维度 + 匹配进度联动 + 平滑动画
java·服务器·前端
Yeniden2 小时前
Deepeek用大白话讲解 --> 迭代器模式(企业级场景1,多种遍历方式2,隐藏集合结构3,Java集合框架4)
java·开发语言·迭代器模式