首先,使用简单的for循环时,list.remove()肯定是没问题的,只要注意一下下标别越界就行。
示例代码:
List<String> ll = new ArrayList<String>();
ll.add("1");
ll.add("2");
ll.add("3");
ll.add("4");
ll.add("5");
for(int i=0;i<5;i++){
if(i==2){
ll.remove(i);
}
}
for (String s : ll) {
System.out.println(s);
}
运行结果 1 2 4 5
但是这种方法不能用于删除多个元素,如下代码:
List<String> strlist = new ArrayList<String>(); strlist.add("1"); strlist.add("2"); strlist.add("3"); strlist.add("4"); strlist.add("5"); strlist.add("6"); for(int i = 0 ;i< 5;i++){ if(i == 1 || i ==3){ strlist.remove(i); } } for(String s : strlist){ System.out.println(s); }
运行结果是 1 3 4 6
因为元素删除后,后边元素前移了,删除的第四个元素,实际上是原来的第五个元素:5。
还有一种情况,也容易犯浑:
List<String> strlist = new ArrayList<String>(); strlist.add("1"); strlist.add("2"); strlist.add("2"); strlist.add("3"); strlist.add("4"); strlist.add("5"); for(int i = 0 ;i< 5;i++){ if("2".equals(strlist.get(i))){ strlist.remove(i); } } for(String s : strlist){ System.out.println(s); }
该段代码运行结果, 1 2 3 4 5 ,因为元素删除后前移,正好下次循环时把第二个‘2’漏掉了。
报ConcurrentModificationException异常的情况一般是因为在使用了迭代器iterator的情况下,没使用iterator.remove(),而是使用list.remove()造成的,推荐做法如下:
示例代码:
1 //准备数据 2 List<Student> list = new ArrayList<>(); 3 list.add(new Student("1")); 4 list.add(new Student("2")); 5 list.add(new Student("3")); 6 list.add(new Student("4")); 7 8 //遍历删除 9 Iterator<Student> iterator = list.iterator(); 10 while (iterator.hasNext()) { 11 Student student = iterator.next(); 12 if ("1".equals(student.getId())) { 13 iterator.remove();//使用迭代器的删除方法删除 14 } 15 }
在使用foreach遍历时,foreach内部实现实际上还是使用了迭代器,所以会报错。阅读源码可以发现,在删除倒数第二个元素时不会报错,具体原因可以看:https://blog.csdn.net/bimuyulaila/article/details/52088124
另:调用Arrays.asList()生产的List不可以使用add,remove,这是由Arrays.asList() 返回的市Arrays的内部类ArrayList, 而不是java.util.ArrayList。Arrays的内部类ArrayList和java.util.ArrayList都是继承AbstractList,remove、add等方法AbstractList中是默认throw UnsupportedOperationException而且不作任何操作。详细介绍