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.小心,谨慎

 

posted @ 2017-09-29 23:49  zY-  阅读(1424)  评论(0编辑  收藏  举报