Java中的fail-fast机制

fail-fast 机制是Java集合(Collection)中的一种错误机制。当多个线程对同一个集合的内容进行操作时,就可能会产生fail-fast事件。

例如:当某一个线程 A 通过 iterator 去遍历某集合的过程中,若该集合的内容被其他线程所改变了,那么线程 A 访问集合时,就会抛出 ConcurrentModificationException 异常,产生 fail-fast 事件。这里的操作主要是指 add、remove 和 clear,对集合元素个数进行修改。

举例代码:

单线程,在foreach循环里对某些集合元素进行元素的remove/add操作的时候,会触发fail-fast机制

public static void main(String[] args){
  List<String> strList = new ArrayList<>();
  strList.add("AA");
  strList.add("aa");
  strList.add("BB");
  strList.add("CC");
  for(String str : strList){
     if("aa".equals(str)){
       strList.remove(str);
     }
  }
}

多线程,在一个线程读时,另一个线程写入list,读线程会fail-fast

// 测试
public class TreadDemo1 {
   public static void main(String[] args){
        List<String> strList = new ArrayList<>();
        strList.add("AA");
        strList.add("aa");
        strList.add("BB");
        strList.add("CC");
        strList.add("DD");
        new MyThread1(strList).start();
        new MyThread2(strList).start();
   }
   static class Mythread1 extends Thread {
       private List<String> list;
       public Mythread1(List<String> list){
             this.list = list;
       }
       @Override
       public void run(){
             for(String str : list){
                 try{
                     Thread.sleep(100);
                 }catch(InterruptedException e){
                     e.printStackTrace();
                 }
                 System.out.println("MtThread1:"+str);
             }
       }
  }
  static class MyThread2 extends Thread {
       private list<String> list;
       public Mythread2(List<String> list){
             this.list = list;
       }
       @Override
       public void run(){
             for(int i = 0; i < list.size(); i++){
                 try{
                     Thread.sleep(100);
                 }catch(InterruptedException e){
                     e.printStackTrace();
                 }
                 if("aa".equals(list.get(i))){
                     list.remove(i);
                 }
             }
             System.out.println("MtThread2:"+list);
       }
  }
}

原理:

将单线程编译后的.class反编译后发现,foreach其实是依赖while循环和Iterator实现的。通过跟踪代码的异常堆栈,发现真正抛出异常的代码是:java.util.ArrayList$Itr.checkForComodification();该方法实在iterator.next()方法中调用的:

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

在该方法中modCount 和 expectedModCount进行了比较,如果二者不相等,则抛出ConcurrentModificationException 异常。

modCount 是ArrayList中的一个成员变量。表示该集合实际被修改的次数。(操作集合类的remove()、add()、clear()方法会改变这个变量值)

expectedModCount 是ArrayList 中的一个内部类---Itr(Iterator接口)中的成员变量。表示这个迭代器预期该集合被修改的次数。其值随着Itr被创建而初始化。只有通过迭代器对集合进行操作,该值才会改变。

所以,在使用Java的集合类的时候,如果发生ConcurrentModificationException 异常,优先考虑fail-fast有关的情况。

解决方式:

1)使用普通for循环进行操作

普通for循环没有使用到Iterator的遍历,所以不会进行fail-fast的检验。

public static void main(String[] args){
        List<String> strList = new ArrayList<>();
        strList.add("AA");
        strList.add("aa");
        strList.add("BB");
        strList.add("CC");
        strList.add("DD");
        for(int i = 0; i < strList.size(); i++){
                 if("aa".equals(strList.get(i))){
                     strList.remove(i);
                 }
        }
   }


 

2)直接使用Iterator 进行操作

public static void main(String[] args){
        List<String> strList = new ArrayList<>();
        strList.add("AA");
        strList.add("aa");
        strList.add("BB");
        strList.add("CC");
        strList.add("DD");
        Iterator<String> iterator = strList.iterator();
        while(iterator.hasNext()){
             if("aa".equals(iterator.next())){
                iterator.remove();
             }
        }
   }


 

3)使用Java 8中提供的filter 过滤

Java 8 中可以把集合转换成流,对于流有一种filter操作,可以对原始Stream 进行某项过滤,通过过滤的元素被留下了生成一个新的Stream。

public static void main(String[] args){
        List<String> strList = new ArrayList<>();
        strList.add("AA");
        strList.add("aa");
        strList.add("BB");
        strList.add("CC");
        strList.add("DD");
        strList = strList.stream().filter(e -> !"aa".equals(e)).collect(Collectors.toList());
        System.out.println(strList);
   }


 

4)使用fail-safe的集合类

为了避免触发fail-fast机制导致异常,我们可以使用Java中提供的一些采用了fail-safe机制的集合类。java.util.concurrent包下的容器都是fail-safe的,可以在多线程下并发使用,并发修改。同时也可以在foreach中进行add/remove等操作。

5)也可以使用foreach循环

如果我们非常确定一个集合中,某个即将删除的元素只包含一个的话,也是可以使用foreach循环的,只要删除之后,立即结束循环体,不在继续执行遍历就可以。

public static void main(String[] args){
  List<String> strList = new ArrayList<>();
  strList.add("AA");
  strList.add("aa");
  strList.add("BB");
  strList.add("CC");
  for(String str : strList){
     if("aa".equals(str)){
       strList.remove(str);
       break;
     }
  }
  System.out.println(strList);
}

如果觉得本文对您有帮助,欢迎点赞+收藏+关注!

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
1. Vector的组合操作是线程安全的。Vector是通过在每个方法上添加synchronized关键字来实现线程安全的。这意味着同一时间只能有一个线程访问Vector的方法,从而避免了多线程并发访问导致的数据不一致性问题。 2. 要实现一个复合型的操作,可以使用事务来保证一组操作的原子性。在Java,可以使用数据库的事务来实现复合型操作,或者使用编程语言提供的事务管理器。 3. ArrayList进行序列化和反序列化可以使用Java的序列化机制。通过实现Serializable接口,在需要序列化和反序列化的ArrayList上添加关键字transient,并提供自定义的readObject和writeObject方法来控制序列化和反序列化的过程。 4. 如果已有一个ArrayList想得到一个线程安全的List,可以使用Collections类的synchronizedList方法。该方法返回一个线程安全的List,它使用synchronized关键字来保证方法的同步访问。 5. synchronizedList和Vector都实现了List接口并提供了线程安全的操作。它们的主要区别在于线程安全的实现方式不同。synchronizedList使用了内部对象锁来保证方法的同步访问,而Vector在每个方法上使用了synchronized关键字来实现同步访问。 6. 在遍历时同时删除元素,不能直接使用foreach循环或普通for循环来删除元素,因为会引发ConcurrentModificationException异常。可以使用迭代器的remove方法来删除元素,或者使用ListIterator的remove方法。这是因为在遍历过程使用迭代器或ListIterator的remove方法,会更新迭代器的状态并保持一致性,符合fail-fast机制。 7. HashMap的数据结构是哈希表(hash table)。底层存储是一个数组,每个数组元素对应一个链表或红黑树。通过计算键的哈希值,将键值对映射到数组的对应位置,解决哈希冲突采用链表或红黑树。 8. 使用数组+列表的主要目的是为了兼顾数组和列表的优点。数组在访问元素时效率高,而列表在动态添加和删除元素时效率高。通过使用数组+列表的结构,可以充分利用两者的优点,同时满足不同的操作需求。 9. hashCode方法用于计算对象的哈希码,equals方法用于判断两个对象是否相等。hashCode方法和equals方法是配套使用的,当两个对象的hashCode相等并且equals方法返回true时,它们被认为是相等的对象。 10. equals方法可以重写。重写equals方法的目的是根据业务需求重新定义两个对象是否相等的逻辑。一般需要重写equals方法时,也需要同时重写hashCode方法以保持一致性。 11. IDEA生成equals方法有多种方法,可以通过快捷键Alt+Insert,选择"equals() and hashCode()"来自动生成equals方法。也可以通过右键菜单选择"Generate",然后选择"equals() and hashCode()"来生成equals方法。 12. toString方法的实现有多种方法。可以手动实现toString方法,在方法返回对象的字符串表示。也可以使用第三方库,如Apache Commons Lang的ToStringBuilder或Guava的ToStringHelper来简化toString方法的实现。 13. Java的字符串拼接有多种方式,包括使用"+"操作符、使用StringBuilder或StringBuffer的append方法、使用字符串模板(如String.format)等。 14. "+"操作符是基于StringBuilder或StringBuffer实现的。在编译时,Java编译器会对字符串拼接表达式进行优化,将其转换为使用StringBuilder或StringBuffer的append方法来实现字符串拼接。 15. Java的字符串是不可变的。final关键字修饰的类确保该类不可被继承,而不是字符串本身的可变性。不可变的字符串意味着一旦创建就不能修改其值,任何修改字符串的操作都会创建一个新的字符串对象。 16. Java没有固定的长度限制。根据具体的实现和系统资源限制,字符串的长度可能有一定的限制,但一般情况下可以存储非常大的字符串。 17. 哈希算法常见的有MD5、SHA-1、SHA-256等。哈希冲突可以通过增加哈希表的长度、使用更好的哈希函数、使用链表或红黑树等方式来解决。其,常见的解决哈希冲突的方法有开放寻址法和链地址法。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值