首先,使用简单的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而且不作任何操作。详细介绍