印象中循环删除list中的元素使用for循环的方式是有问题的,但是可以使用增强的for循环,然后今天在使用时发现报错了,然后去科普了一下,再然后发现这是一个误区。

1、for循环遍历list删除元素

for(int i=0;i<list.size();i++){
     if(list.get(i).equals("del"))
     list.remove(i);
}

 

这种方式的问题在于,删除某个元素后,list的大小发生了变化,而你的索引也在变化,所以会导致你在遍历的时候漏掉某些元素。比如当你删除第1个元素后,继续根据索引访问第2个元素时,因为删除的关系后面的元素都往前移动了一位,所以实际访问的是第3个元素。因此,这种方式可以用在删除特定的一个元素时使用,但不适合循环删除多个元素时使用。

 

2、增强for循环删除链表元素

for(String x:list){
if(x.equals("del"))
list.remove(x);
}

 

ConcurrentModificationException,因为元素在使用的时候发生了并发的修改,导致异常抛出。但是删除完毕马上使用break跳出,则不会触发报错。

 

3、iterator遍历

 

Iterator<String> it = list.iterator();
while(it.hasNext()){
String x = it.next();
if(x.equals("del")){
it.remove();
}
}

 

 

  这种方式可以正常的循环及删除。但要注意的是,使用iterator的remove方法,如果用list的remove方法同样会报上面提到的ConcurrentModificationException错误。

 

  总结:

         (3)推荐迭代器删除

 

附: 方法一:使用 Java    8 中提供的 filter 过滤,Java    8 中可以把集合转换成流,对于流有一种 filter 操作,    可以对原始    Stream    
        进行某项测试,通过测试的元素被留下来生成一个新    Stream。
       

List<Integer> list = new LinkedList<>();
        for (int i = 0; i < 10; i++){
            list.add(i);
        }
        List<Integer> result = list.stream().filter(i -> i != 1).collect(Collectors.toList());
        System.out.println(result);

      方法二:在Java中,除了一些普通的集合类以外,还有一些采用了fail-safe机制的集合类。这样的集合容器在遍历时不是直接在集合内容上访问的,而是先复制原有集合
内容,在拷贝的集合上进行遍历。由于迭代时是对原集合的拷贝进行遍历,所以在遍历过程中对原集合所作的修改并不能被迭代器检测到,所以不会触发 ConcurrentModificationException。

      

ConcurrentLinkedDeque<String> userNames = new ConcurrentLinkedDeque<String>() {{
            add("Hollis");    add("hollis");    add("HollisChuang");    add("H"); add("Hollis");
        }};

        for (String userName : userNames) {
            if (userName.equals("Hollis")) {
                userNames.remove();
            }
        }
        System.out.println(userNames);