(简单易懂)Java的快速失败(fail-fast)与安全失败,源码分析+详细讲解
之前在程序中遇到快速失败的问题,在网上找解释时发现网上的问题总结都比较片面,故打算自己总结一个,也可以供以后参考。
--宇的季节
首先什么是快速失败?
快速失败是为了提示程序员在多线程的情况下不要用线程不安全的集合(bug)的一种机制。
当然在单线程的情况下有时也会出现ConcurrentModificationException异常,下面我们就根据ArrayList来探索快速失败的内部实现方法。
示例一:
public static void main(String[] args) {
List<String> list = new ArrayList<String>();
//为集合添加100个数
for (int i = 0; i < 100; i++)
list.add(i+"");
//获取集合的迭代器
Iterator<String> iter = list.iterator();
//我们在集合中删除一个数据,注意我们是直接用集合操作而不是迭代器
list.remove("2");
//再使用迭代器获取数据
iter.next();
}
执行结果:
果不其然,报了ConcurrentModificationException的错误。
为什么会报错呢?下面我们就来看看ArrayList的源码(为了方便阅读,把没用的都省略了)
- public class ArrayList<E> ,,,,,
- {
- ....
- //modCount:集合的修改次数
- protected transient int modCount = 0;
- //当出现增删操作时,便会modCount++;
- public E remove(int index) {
- ....
- modCount++;
- ......
- }
- ....
- //通过此方法获取迭代器
- public Iterator<E> iterator() {
- return new Itr();
- }
- ....
- //迭代器是ArrayList的一个内部类
- private class Itr implements Iterator<E> {
- ...
- //每次获取一个迭代器,就新建一个对象,将expectedModCount赋值为modCount
- int expectedModCount = modCount;
- .....
- @SuppressWarnings("unchecked")
- public E next() {
- //检查expectedModCount和modCount是否相等
- checkForComodification();
- .....
- }
- //这个就是检查的方法
- final void checkForComodification() {
- if (modCount != expectedModCount)
- throw new ConcurrentModificationException();
- }
- .....
- }
我们已经了解了ArrayList的大概结构,那么示例一的报错也就不难解释了:
假设创建一个迭代器时modCount=10,那么expectedModCount=10(注意,modCount是ArrayList的参数,而expectedModCount是迭代器的参数,从创建迭代器之后,他们就没联系了)
当执行list.remove("2");操作时,modCount++=11;而此时expectedModCount=10;当我们再执行iter.next()时,很明显两个数不一致,抛出异常。
结论:当我们在获取一个快速失败集合(只要不是concurrent包下的集合都是快速失败的)的迭代器时候,之后在操作迭代器的中间不能对集合本身再进行增删操作,否则会抛异常。
但是,快速失败机制的意义并不在此,上面已经说到了,快速失败是为了警告程序员不能在多线程情况下使用线程不安全的集合。
示例2:
- public class test {
- private static List<String> list = new ArrayList<String>();
- public static void main(String[] args) {
- for (int i = 0; i < 100; i++)
- list.add(i+"");
- System.out.println("线程开始了");
- new threadOne().start();
- new threadTwo().start();
- }
- //线程1每隔10ms遍历一次
- private static class threadOne extends Thread{
- public void run(){
- Iterator<String> iter = list.iterator();
- while(iter.hasNext()){
- System.out.println("线程1遍历到了:"+iter.next());
- try {
- Thread.sleep(1);
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- }
- }
- }
- //线程1每隔10ms删除一个
- private static class threadTwo extends Thread{
- public void run(){
- Iterator<String> iter = list.iterator();
- while(iter.hasNext()){
- System.out.println("线程2删除了:"+iter.next());
- iter.remove();
- try {
- Thread.sleep(1);
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- }
- }
- }
- }
执行结果:
在示例2中,我们并没有在使用迭代器中间直接对集合进行增删。但是依然报错。
原因就是因为线程的异步性!
这次只看迭代器的remove()操作即可:
可以看到,两个值的改变被分成了两条程序语句,那么就会出现这种情况:
线程1删除时执行到语句1,modCount+1。
由于语句2还没执行,expectedModCount不会改变
这时处理器跳转到线程2执行。
线程2执行next()时,对比modCount和expectedModCount。
发现两值不一致,抛出ConcurrentModificationException
终于把快速失败搞明白了!那么什么是安全失败呢?
这个就简单了,在使用安全失败的迭代器时,会copy一个集合,之后的操作都是对这个copy的“副本”进行操作。
那么就算是多线程,操作的也是创建迭代器时copy的单独的副本。
所以不会造成冲突,ConcurrentModificationException也就不复存在啦~
(简单易懂)Java的快速失败(fail-fast)与安全失败,源码分析+详细讲解的更多相关文章
- Java显式锁学习总结之六:Condition源码分析
概述 先来回顾一下java中的等待/通知机制 我们有时会遇到这样的场景:线程A执行到某个点的时候,因为某个条件condition不满足,需要线程A暂停:等到线程B修改了条件condition,使con ...
- Java显式锁学习总结之五:ReentrantReadWriteLock源码分析
概述 我们在介绍AbstractQueuedSynchronizer的时候介绍过,AQS支持独占式同步状态获取/释放.共享式同步状态获取/释放两种模式,对应的典型应用分别是ReentrantLock和 ...
- Java并发包中Semaphore的工作原理、源码分析及使用示例
1. 信号量Semaphore的介绍 我们以一个停车场运作为例来说明信号量的作用.假设停车场只有三个车位,一开始三个车位都是空的.这时如果同时来了三辆车,看门人允许其中它们进入进入,然后放下车拦.以后 ...
- Java显式锁学习总结之四:ReentrantLock源码分析
概述 ReentrantLock,即重入锁,是一个和synchronized关键字等价的,支持线程重入的互斥锁.只是在synchronized已有功能基础上添加了一些扩展功能. 除了支持可中断获取锁. ...
- 转!!Java学习之自动装箱和自动拆箱源码分析
自动装箱(boxing)和自动拆箱(unboxing) 首先了解下Java的四类八种基本数据类型 基本类型 占用空间(Byte) 表示范围 包装器类型 boolean 1/8 true|fal ...
- Java学习之自动装箱和自动拆箱源码分析
自动装箱(boxing)和自动拆箱(unboxing) 首先了解下Java的四类八种基本数据类型 基本类型 占用空间(Byte) 表示范围 包装器类型 boolean 1/8 true|false ...
- 【Java】CAS的乐观锁实现之AtomicInteger源码分析
1. 悲观锁与乐观锁 我们都知道,cpu是时分复用的,也就是把cpu的时间片,分配给不同的thread/process轮流执行,时间片与时间片之间,需要进行cpu切换,也就是会发生进程的切换.切换涉及 ...
- Java - "JUC线程池" 线程状态与拒绝策略源码分析
Java多线程系列--“JUC线程池”04之 线程池原理(三) 本章介绍线程池的生命周期.在"Java多线程系列--“基础篇”01之 基本概念"中,我们介绍过,线程有5种状态:新建 ...
- Java 自动装箱、拆箱机制及部分源码分析
Integer i = 10; //装箱,反编译后发现调用Integer.valueOf(int i) int t = i; //拆箱,反编译后发现调用i.intValue() public clas ...
随机推荐
- Klass与Oop
前段时间,一直在看<Hotspot实战>,顺便编译了一份OpenJDK的源码,然后就在eclipse里面调试起来. 虽然我的入门语言是c/c++,但是被Java拉过来好几年了,现在再看源码 ...
- SQL Server系列之SQL Server 2016 中文企业版详细安装步骤(超多图)
1. 下载地址 下载地址 :https://www.microsoft.com/en-us/server-cloud/products/sql-server-2016/ 官方技术文档:https:// ...
- LuaFramework热更新过程(及可更新的loading界面实现)
1.名词解释: 资源包:点击 LuaFramework | Build XXX(平台名) Resource,框架会自动将自定义指定的资源打包到StreamingAssets文件夹,这个 ...
- [JAVASCRIPT]实现页面复制至电脑剪贴板
一. 方法 方1: window.clipboarddata 可惜不支持chrome , chrome 下会提示找不到 clipboarddata 对象 方2: 采用国外大牛写的ZeroClipbo ...
- app中rem算法
第一次用vue做APP被rem坑惨了 下面贴出 rem 的算法及使用方法 在自定义js中定义函数 export default { install: function(Vue, options) { ...
- javascript中this的用法
this是Javascript语言的一个关键字. 它代表函数运行时,自动生成的一个内部对象,只能在函数内部使用.比如, function test(){ this.x = 1; } 随着函数使用场合的 ...
- Java单线程文件下载,支持断点续传功能
前言: 程序下载文件时,有时会因为各种各样的原因下载中断,对于小文件来说影响不大,可以快速重新下载,但是下载大文件时,就会耗费很长时间,所以断点续传功能对于大文件很有必要. 文件下载的断点续传: 1. ...
- 微信小程序框架探究和解析
何为框架 你对微信小程序的技术框架了解多少? 对wepy 框架进行一系列的深入了解 微信小程序框架解析和探究 小程序组件化框架WePY 在性能调优上做出的探究 开发者培训班上海专场PPT分享:小程序框 ...
- Zend Framework1 框架入门(针对Windows,包含安装配置与数据库增删改查)
最近公司接的项目需要用到Zend Framework框架,本来需要用的是ZendFramework2 ,但是由于原有代码使用了ZendFramework1 框架,所以顺带学习了.现将一些基础入门记录一 ...
- Redis-aof持久化
什么是redis的aof? aof 是 appendonly file 的缩写, 是redis系统提供的一种记录redis操作的持久化方案, 在aof生成的文件中, 将记录发生在redis的操作, 从 ...