0
点赞
收藏
分享

微信扫一扫

[Fast-Fail] Java快速失败机制


前言:
学号java基础,我们才能在开发中游刃有余的解决问题,才会减少BUG的出现几率。

产生原因分析

​fail-fast机制是java集合(Collection)中的一种错误机制。​​​ 当多个线程对同一个集合的内容进行操作时,就可能会产生fail-fast机制。
注意:迭代器的fail-fast机制并不能被保证一定会发生,一般来说,在存在不同步的并发修改时不可能做出任何有力保证,但是fail-fast会尽最大努力抛出ConcurrentModificationException错误。因此如果一个程序依赖这个异常去保证其运行的正确性是错误的,快速失败机制只能用来标识这个错误。

fail-fast原理

产生fail-fast事件,是通过抛出ConcurrentModificationException异常来触发的。这个异常是在操作Iterator时抛出的异常。我们先看看Iterator的源码。

public abstract class AbstractList<E> extends AbstractCollection<E> implements List<E> {

...

// AbstractList中唯一的属性
// 用来记录List修改的次数:每修改一次(添加/删除等操作),将modCount+1
protected transient int modCount = 0;

// 返回List对应迭代器。实际上,是返回Itr对象。
public Iterator<E> iterator() {
return new Itr();
}

// Itr是Iterator(迭代器)的实现类
private class Itr implements Iterator<E> {
int cursor = 0;

int lastRet = -1;

// 修改数的记录值。
// 每次新建Itr()对象时,都会保存新建该对象时对应的modCount;
// 以后每次遍历List中的元素的时候,都会比较expectedModCount和modCount是否相等;
// 若不相等,则抛出ConcurrentModificationException异常,产生fail-fast事件。
int expectedModCount = modCount;

public boolean hasNext() {
return cursor != size();
}

public E next() {
// 获取下一个元素之前,都会判断“新建Itr对象时保存的modCount”和“当前的modCount”是否相等;
// 若不相等,则抛出ConcurrentModificationException异常,产生fail-fast事件。
checkForComodification();
try {
E next = get(cursor);
lastRet = cursor++;
return next;
} catch (IndexOutOfBoundsException e) {
checkForComodification();
throw new NoSuchElementException();
}
}

public void remove() {
if (lastRet == -1)
throw new IllegalStateException();
checkForComodification();

try {
AbstractList.this.remove(lastRet);
if (lastRet < cursor)
cursor--;
lastRet = -1;
expectedModCount = modCount;
} catch (IndexOutOfBoundsException e) {
throw new ConcurrentModificationException();
}
}

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

...
}

从这段代码中我们可以看出,我们在调用next和remove方法时,都会执行checkForComodification。若"modCount 不等于 expectedModCount"。则抛出ConcurrentModificationException异常,产生fail-fast事件。
要搞明白 fail-fast机制,我们就要需要理解什么时候“modCount 不等于 expectedModCount”!
从Itr类中,我们知道 expectedModCount 在创建Itr对象时,被赋值为 modCount。通过Itr,我们知道:expectedModCount不可能被修改为不等于 modCount。所以,需要考证的就是modCount何时会被修改。
然后我们查看ArrayList源码,发现无论是add(),remove(),clear() ,只要涉及到修改集合中的元素个数,都会改变modCount的值。

分析完了,我们来最终总结一下:
新建了一个ArrayList,名称为arrayList, 向arrayList中添加内容。新建一个“线程a”,并在“线程a”中通过Iterator反复的读取arrayList的值。新建一个“线程b”,在“线程b”中删除arrayList中的一个“节点A”。
在某一时刻,“线程a”创建了arrayList的Iterator。此时“节点A”仍然存在于arrayList中,创建arrayList时,expectedModCount = modCount(假设它们此时的值为N)。
在“线程a”在遍历arrayList过程中的某一时刻,“线程b”执行了,并且“线程b”删除了arrayList中的“节点A”。“线程b”执行remove()进行删除操作时,在remove()中执行了“modCount++”,此时modCount变成了N+1!
“线程a”接着遍历,当它执行到next()函数时,调用checkForComodification()比较“expectedModCount”和“modCount”的大小;而“expectedModCount=N”,“modCount=N+1”,这样,便抛出ConcurrentModificationException异常,产生fail-fast事件。
​​​即,当多个线程对同一个集合进行操作的时候,某线程访问集合的过程中,该集合的内容被其他线程所改变(即其它线程通过add、remove、clear等方法,改变了modCount的值);这时,就会抛出ConcurrentModificationException异常,产生fail-fast事件。​

fail-fast解决办法

若在多线程环境下使用fail-fast机制的集合,建议使用“java.util.concurrent包下的类”去取代“java.util包下的类”。


举报

相关推荐

0 条评论