印象中循環(huán)刪除list中的元素使用for循環(huán)的方式是有問題的,但是可以使用增強的for循環(huán),然后今天在使用時發(fā)現(xiàn)報錯了,然后去科普了一下,再然后發(fā)現(xiàn)這是一個誤區(qū)。下面就來講一講。。伸手黨可直接跳至文末??纯偨Y(jié)。。
JAVA中循環(huán)遍歷list有三種方式for循環(huán)、增強for循環(huán)(也就是常說的foreach循環(huán))、iterator遍歷。
1、for循環(huán)遍歷list
//1,普通for循環(huán)刪除,索引要--
for(int i = 0; i < list2.size(); i++) {
if("b".equals(list2.get(i))) {
list2.remove(i);//通過索引刪除元素
/當刪除索引為1的b時,此刻后面的索引全部向前走第二個索引為2的b向前一步他的索引即變?yōu)?
//此刻i的值為1,執(zhí)行后i++變?yōu)?,即后來索引為1的b不被判斷
//應(yīng)該list2.remove(i--);
}
}
這種方式的問題在于,刪除某個元素后,list的大小發(fā)生了變化,而你的索引也在變化,所以會導(dǎo)致你在遍歷的時候漏掉某些元素。比如當你刪除第1個元素后,繼續(xù)根據(jù)索引訪問第2個元素時,因為刪除的關(guān)系后面的元素都往前移動了一位,所以實際訪問的是第3個元素。因此,這種方式可以用在刪除特定的一個元素時使用,但不適合循環(huán)刪除多個元素時使用。
2、增強for循環(huán)
for(String x:list){ if(x.equals("del"))
list.remove(x);
}
這種方式的問題在于,刪除元素后繼續(xù)循環(huán)會報錯誤信息ConcurrentModificationException,因為元素在使用的時候發(fā)生了并發(fā)的修改,導(dǎo)致異常拋出。但是刪除完畢馬上使用break跳出,則不會觸發(fā)報錯。
3、iterator遍歷
//2,迭代器刪除
Iterator<String> it = list2.iterator();
while(it.hasNext()) {
if("e".equals(it.next())) {
//list2.remove("e");
//不能用集合的刪除方法,因為迭代過程中如果集合修改會出現(xiàn)并發(fā)修改異常
// java.util.ConcurrentModificationException
it.remove();
}
}
System.out.println(list2);//[a, b, c, d, f, g]
//第二種
for(Iterator<String> it2 = list2.iterator(); it2.hasNext();) {
if("f".equals(it2.next())) {
//list.remove("f");//不能用集合的刪除方法,因為迭代過程中如果集合修改會出現(xiàn)并發(fā)修改異常
it2.remove();
}
}
這種方式可以正常的循環(huán)及刪除。但要注意的是,使用iterator的remove方法,如果用list的remove方法同樣會報上面提到的ConcurrentModificationException錯誤。
總結(jié):
(1)循環(huán)刪除list中特定一個元素的,可以使用三種方式中的任意一種,但在使用中要注意上面分析的各個問題。
(2)循環(huán)刪除list中多個元素的,應(yīng)該使用迭代器iterator方式。
(3)注意事項(重要):http://www.itdecent.cn/p/445de4ba6f35