zoukankan      html  css  js  c++  java
  • ArrayList : ArrayList ConcurrentModificationException (转)

    1.ConcurrentModificationException

    ConcurrentModificationException 出现在使用 ForEach遍历,迭代器遍历的同时,进行删除,增加出现的异常。平常使用的ArrayList, HashMap都有可能抛出这种异常,粗心的话,很容易犯这种错误,导致线上事故!

    2. 情景列举

    下面就ArrayList的一些使用场景,来讨论是否会抛出ConcurrentModificationException

    2.1 For..i 遍历 

    这个遍历的意思,是指 for(int i = 0 ; i <list.size(); i ++) 这种使用下标进行遍历的方式。

    这种情形下,增加都不会有 ConcurrentModificationException。但是也可能导致另外的一些问题,比如下面这段代码,会死循环

    (代码手打,可能有错误)

    复制代码
    List<Integer> list = new Arraylist<>();
    list.add(1);
    list.add(2);
    list.add(3);
    for(int i = 0;i<list.size();i++){
       list.add(i);  
    }
    复制代码

    遍历删除的情况下,不会有ConcurrentModificationException,但是要注意代码,防止数组越界异常。下面这种形式的代码会抛出数组越界异常。

    复制代码
    List<Integer> list = new ArrayList<>();
    list.add(1);
    list.add(2);
    list.add(3);
    int length = list.size();
    for(int i = 0;i<length;i++){
          list.remove(i);
    }
    复制代码

    当然正常情况下,我们不会先计算 list.size(),而是直接在循环里使用,i<list.size()。这时候可能会导致另外一个问题。看代码

    复制代码
    List<Integer> list = new ArrayList<>();
    list.add(1);
    list.add(1);
    list.add(1);
    list.add(1);
    for(int i = o;i<list.size();i++){
       if(list.get(i) == 1){
             list.remove(i);
          //这里加上 i-- 就没有问题了 }
    //list.size == 2 也就是说还有两个元素没有删除 }
    复制代码

    2.1ForEach 遍历

    ForEach 遍历就是 For(Object o : List<Object>) 这种遍历方式,众所周知,ForEach循环只是JAVA的一个语法糖,在字节码层面上,等同于迭代器循环遍历。在这种情形下,增加元素一定会抛出ConcurrentModificationException,

    而删除元素在大多数情况下,会抛出ConcurrentModificationException(小知识,当且仅当删除小标为 size()-2,也就是倒数第二个元素的时候,不会抛出异常)。

    这种情况下,会有异常抛出

    复制代码
     List<Integer> list = new ArrayList<>();
            list.add(1);
            list.add(2);
            list.add(3);
            for (Integer i : list) {
                if(i == 1){
                    list.remove(i);
                }
          }
    复制代码

    可以修改上面的判断语句, i == 1 修改为 i == 2 则不会抛出异常。

    3如何避免ConcurrentModificationException

    1. 需要遍历新增时,最好new一个和老List相同的临时List,遍历老的List,然后在临时List上进行元素的增加

    2. 需要进行删除时,使用迭代器删除(iterator.remove()),而不是直接调用 list.remove()

    3.小心,谨慎

    转自: https://www.cnblogs.com/mezhouyang/p/7613262.html

  • 相关阅读:
    DROP TABLE 恢复【一】
    Recover InnoDB dictionary
    Percona XtraDB Cluster
    主从复制延时判断
    Keepalived+MySQL实现高可用
    Performance Tuning MySQL
    Redis实现异步消息队列与延时队列
    Python多线程中的setDaemon
    Python实现远程控制单片机led状态
    【机器学习】朴素贝叶斯应用实例
  • 原文地址:https://www.cnblogs.com/yongdaimi/p/12957813.html
Copyright © 2011-2022 走看看