java fail fast_Java学习笔记(40)——Java集合12之fail-fast

一、fail-fast概述

“快速失败”也就是fail-fast,它是Java集合的一种错误检测机制。当多个线程对集合进行结构上的改变的操作时,有可能会产生fail-fast机制。记住是有可能,而不是一定。例如:当某一个线程A通过iterator去遍历某集合的过程中,若该集合的内容被其他线程所改变了;那么线程A访问集合时,就会抛出ConcurrentModificationException异常,产生fail-fast事件。

在详细介绍fail-fast机制的原理之前,先通过一个示例来认识fail-fast。

示例代码:(FastFailTest.java)

import java.util.*;

import java.util.concurrent.*;

/*

* @desc java集合中Fast-Fail的测试程序。

*

*   fast-fail事件产生的条件:当多个线程对Collection进行操作时,若其中某一个线程通过iterator去遍历集合时,该集合的内容被其他线程所改变;则会抛出ConcurrentModificationException异常。

*   fast-fail解决办法:通过util.concurrent集合包下的相应类去处理,则不会产生fast-fail事件。

*

*   本例中,分别测试ArrayList和CopyOnWriteArrayList这两种情况。ArrayList会产生fast-fail事件,而CopyOnWriteArrayList不会产生fast-fail事件。

*   (01) 使用ArrayList时,会产生fast-fail事件,抛出ConcurrentModificationException异常;定义如下:

*            private static List list = new ArrayList();

*   (02) 使用时CopyOnWriteArrayList,不会产生fast-fail事件;定义如下:

*            private static List list = new CopyOnWriteArrayList();

*

* @author skywang

*/

public class FastFailTest {

private static List list = new ArrayList();

//private static List list = new CopyOnWriteArrayList();

public static void main(String[] args) {

// 同时启动两个线程对list进行操作!

new ThreadOne().start();

new ThreadTwo().start();

}

private static void printAll() {

System.out.println("");

String value = null;

Iterator iter = list.iterator();

while(iter.hasNext()) {

value = (String)iter.next();

System.out.print(value+", ");

}

}

/**

* 向list中依次添加0,1,2,3,4,5,每添加一个数之后,就通过printAll()遍历整个list

*/

private static class ThreadOne extends Thread {

public void run() {

int i = 0;

while (i<6) {

list.add(String.valueOf(i));

printAll();

i++;

}

}

}

/**

* 向list中依次添加10,11,12,13,14,15,每添加一个数之后,就通过printAll()遍历整个list

*/

private static class ThreadTwo extends Thread {

public void run() {

int i = 10;

while (i<16) {

list.add(String.valueOf(i));

printAll();

i++;

}

}

}

}

运行结果:

运行该代码,抛出异常java.util.ConcurrentModificationException!即,产生fail-fast事件!

结果说明:

(01) FastFailTest中通过 new ThreadOne().start() 和 new ThreadTwo().start() 同时启动两个线程去操作list。

ThreadOne线程:向list中依次添加0,1,2,3,4,5。每添加一个数之后,就通过printAll()遍历整个list。

ThreadTwo线程:向list中依次添加10,11,12,13,14,15。每添加一个数之后,就通过printAll()遍历整个list。

(02) 当某一个线程遍历list的过程中,list的内容被另外一个线程所改变了;就会抛出ConcurrentModificationException异常,产生fail-fast事件。

二、fail-fast解决方案

fail-fast机制,是一种错误检测机制。它只能被用来检测错误,因为JDK并不保证fail-fast机制一定会发生。若在多线程环境下使用fail-fast机制的集合,建议使用“java.util.concurrent包下的类”去取代“java.util包下的类”。

所以,本例中只需要将ArrayList替换成java.util.concurrent包下对应的类即可。

即,将代码

private static List list = new ArrayList();

替换为

private static List list = new ArrayList();

则可以解决该办法。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值