fail-fast机制简介

什么是fail-fast

fail-fast 机制是java集合(Collection)中的一种错误机制。它只能被用来检测错误,因为JDK并不保证fail-fast机制一定会发生。当多个线程对同一个集合的内容进行操作时,就可能会产生fail-fast事件。例如:当某一个线程A通过iterator去遍历某集合的过程中,若该集合的内容被其他线程所改变了;那么线程A访问集合时,就会抛出ConcurrentModificationException异常,产生fail-fast事件。

这种“ 及时失败” 的迭代器井不是一种完备的处理机制,而只是“ 善意地” 捕获并发错误,因此只能作为并发问题的预警指示器。

fail-fast示例

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
public class FailFastTest {

    static final List<Integer> list = new ArrayList<>();

    public static void main(String[] args) {
ExecutorService exec = Executors.newCachedThreadPool();
for (int i = 0; i < 5; i++) {
exec.execute(() -> {
add((int)(Math.random() * 10));
print();
});
}
} private static void add(int number) {
list.add(number);
} private static void print() {
Iterator iterator = list.iterator();
while (iterator.hasNext()) {
System.out.println(iterator.next());
}
}
}

以上代码会抛出java.util.ConcurrentModificationException异常

fail-fast的解决办法

使用java.util.concurrent.*下的工具类

深入ArrayList源码看fast-fail的原理

先放上
首先看下ArrayList中的内部类Itr的域

1
2
3
4
5
private class Itr implements Iterator<E> {
int cursor; //1
int lastRet = -1; //2
int expectedModCount = modCount; //3
}
  1. 表示迭代器下一个元素的索引;
  2. 表示迭代器上一个元素的索引;
  3. 在创建一个迭代器时,将当前ArrayList的修改次数赋值给expectedModCount保存。

在上面的示例中,我们可以看到一般的迭代过程是

1
2
3
4
Iterator iterator = list.iterator();
while (iterator.hasNext()) {
System.out.println(iterator.next());
}

分别看下iterator()hasNext()next()三个方法:

  • iterator():没有做任何处理,不过构造时三个域会进行初始化
1
Itr() {}
  • hasNext():判断下一个元素索引是否等于ArrayList的大小,等于说明没有元素了
1
2
3
public boolean hasNext() {
return cursor != size;
}
  • 接下来重点看next()方法
1
2
3
4
5
6
7
8
9
10
11
public E next() {
checkForComodification(); //1
int i = cursor; //2
if (i >= size)
throw new NoSuchElementException();
Object[] elementData = ArrayList.this.elementData;
if (i >= elementData.length)
throw new ConcurrentModificationException();
cursor = i + 1;
return (E) elementData[lastRet = i];
}

在获取下一个元素之前,先调用checkForComodification()进行了检查,检查当前集合的修改次数是不是跟之前保存的相同,如果相同则表示没有被其他线程修改,

1
2
3
4
final void checkForComodification() {
if (modCount != expectedModCount)
throw new ConcurrentModificationException();
}

modCount:modCount 是 AbstractList 的属性值:`protected transient int modCount = 0; 他是一个修改次数计数器,实例化一个集合之后,每次修改(源码的注释成为结构性修改),比如set,add,clear等,计数器都会加1。

这里其实就是fail-fast机制的实现原理了,将修改计数器的变化与容器关联起来:首先在构造迭代器的时候,将当前的修改计数器的值保存,之后进行遍历的时候,每访问一个数据,都要检查当前集合的修改次数是否合法,如果有其他线程修改了集合,那么modCount就会被修改,当前修改计数器的值与之前保存的值(即期望值)不同,那么将抛出ConcurrentModificationException。

fail-fast解决办法
        通过前面的实例、源码分析,我想各位已经基本了解了fail-fast的机制,下面我就产生的原因提出解决方案。这里有两种解决方案:

方案一:在遍历过程中所有涉及到改变modCount值得地方全部加上synchronized或者直接使用Collections.synchronizedList,这样就可以解决。但是不推荐,因为增删造成的同步锁可能会阻塞遍历操作。

方案二:使用CopyOnWriteArrayList来替换ArrayList。推荐使用该方案。

CopyOnWriteArrayList为何物?ArrayList 的一个线程安全的变体,其中所有可变操作(add、set 等等)都是通过对底层数组进行一次新的复制来实现的。 该类产生的开销比较大,但是在两种情况下,它非常适合使用。1:在不能或不想进行同步遍历,但又需要从并发线程中排除冲突时。2:当遍历操作的数量大大超过可变操作的数量时。遇到这两种情况使用CopyOnWriteArrayList来替代ArrayList再适合不过了。

关于CopyOnWriteArrayList请看相关文章

最新文章

  1. mysql去重
  2. TabControl 显示彩色的图示 (XE6 Firemonkey)
  3. DB2 组内分组排序,游标使用
  4. 如何让Form窗体接收KeyDown事件?
  5. ubuntu16.04 64位server安装php7
  6. python处理Excel
  7. php pdo oracle中文乱码
  8. bootstrap 正则表达式
  9. 关于HMTL -[HTML5]
  10. [原创]零基础R语言教程---第一课---认识R语言
  11. 【.NET】Repeater控件简单的数据绑定(有bool,日期,序号)
  12. iframe的优缺点?
  13. Java单线程文件下载,支持断点续传功能
  14. python之路第四篇(基础篇)
  15. ES6笔记
  16. PHP 正则表达式资料
  17. Sandcastle方法生成c#.net帮助类帮助文档chm
  18. NumPy 切片和索引
  19. 网络攻防大作业——用python实现wifi破解
  20. 关于线上js报错问题的思考

热门文章

  1. C++ GUI Qt4学习笔记01
  2. 【微信小程序】使用vscode编写微信小程序项目
  3. 黑客代码HTML
  4. 约数定理(two)
  5. (3.2)狄泰软件学院C++课程学习剖析三
  6. 使用chooseImage上传图片,不压缩,使用原图
  7. 原生javascript兼容性总结
  8. hashcode、equals、HashMap间的关系
  9. linux gsensor驱动分析【转】
  10. leetcode-mid-Linked list-160 Intersection of Two Linked Lists-NO