印象中循環(huán)刪除list中的元素使用for循環(huán)的方式是有問題的礼患,但是可以使用增強(qiáng)的for循環(huán)是钥,然后今天在使用時(shí)發(fā)現(xiàn)報(bào)錯(cuò)了,然后去科普了一下缅叠,再然后發(fā)現(xiàn)這是一個(gè)誤區(qū)悄泥。下面就來講一講。肤粱。伸手黨可直接跳至文末弹囚。看總結(jié)狼犯。余寥。
JAVA中循環(huán)遍歷list有三種方式for循環(huán)领铐、增強(qiáng)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);//通過索引刪除元素
/當(dāng)刪除索引為1的b時(shí),此刻后面的索引全部向前走第二個(gè)索引為2的b向前一步他的索引即變?yōu)?
//此刻i的值為1祝蝠,執(zhí)行后i++變?yōu)?音诈,即后來索引為1的b不被判斷
//應(yīng)該list2.remove(i--);
}
}
這種方式的問題在于,刪除某個(gè)元素后绎狭,list的大小發(fā)生了變化细溅,而你的索引也在變化,所以會(huì)導(dǎo)致你在遍歷的時(shí)候漏掉某些元素儡嘶。比如當(dāng)你刪除第1個(gè)元素后喇聊,繼續(xù)根據(jù)索引訪問第2個(gè)元素時(shí),因?yàn)閯h除的關(guān)系后面的元素都往前移動(dòng)了一位蹦狂,所以實(shí)際訪問的是第3個(gè)元素誓篱。因此,這種方式可以用在刪除特定的一個(gè)元素時(shí)使用凯楔,但不適合循環(huán)刪除多個(gè)元素時(shí)使用窜骄。
2、增強(qiáng)for循環(huán)
for(String x:list){ if(x.equals("del"))
list.remove(x);
}
這種方式的問題在于摆屯,刪除元素后繼續(xù)循環(huán)會(huì)報(bào)錯(cuò)誤信息ConcurrentModificationException邻遏,因?yàn)樵卦谑褂玫臅r(shí)候發(fā)生了并發(fā)的修改,導(dǎo)致異常拋出虐骑。但是刪除完畢馬上使用break跳出准验,則不會(huì)觸發(fā)報(bào)錯(cuò)。
3富弦、iterator遍歷
//2,迭代器刪除
Iterator<String> it = list2.iterator();
while(it.hasNext()) {
if("e".equals(it.next())) {
//list2.remove("e");
//不能用集合的刪除方法,因?yàn)榈^程中如果集合修改會(huì)出現(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");//不能用集合的刪除方法,因?yàn)榈^程中如果集合修改會(huì)出現(xiàn)并發(fā)修改異常
it2.remove();
}
}
這種方式可以正常的循環(huán)及刪除沟娱。但要注意的是,使用iterator的remove方法腕柜,如果用list的remove方法同樣會(huì)報(bào)上面提到的ConcurrentModificationException錯(cuò)誤。
總結(jié):
(1)循環(huán)刪除list中特定一個(gè)元素的矫废,可以使用三種方式中的任意一種盏缤,但在使用中要注意上面分析的各個(gè)問題。
(2)循環(huán)刪除list中多個(gè)元素的蓖扑,應(yīng)該使用迭代器iterator方式唉铜。
(3)注意事項(xiàng)(重要):http://www.reibang.com/p/445de4ba6f35