JAVA中循環遍歷list有三種方式for循環、增強for循環(也就是常說的foreach循環)、iterator遍歷。
1、for循環遍歷list
for(int i=0;i<list.size();i++){ if(list.get(i).equals("del")) list.remove(i); }
這種方式的問題在於,刪除某個元素后,list的大小發生了變化,而你的索引也在變化,所以會導致你在遍歷的時候漏掉某些元素。比如當你刪除第1個元素后,繼續根據索引訪問第2個元素時,因為刪除的關系后面的元素都往前移動了一位,所以實際訪問的是第3個元素。因此,這種方式可以用在刪除特定的一個元素時使用,但不適合循環刪除多個元素時使用。
因為集合的大小是動態變化的,當你刪除一個元素之后,元素中的序號又重新排列,原來第二個應該刪除的元素現在排在了第一個元素的位置,真正刪除的卻是第三個元素,依次類推,刪除的是第一個、第三個、第五個、、、、如果在原來刪除的代碼中加入語句:System.out.println("即將刪除的元素:"+list.get(i));即可驗證。
加入上述語句后輸出的結果:
即將刪除的元素:a
即將刪除的元素:c
即將刪除的元素:e
還剩余的元素個數:2
解決辦法:
究其原因是因為你要刪除的元素往前面移動了,而你的i保存的值依舊往后走,所以如果讓i不往后走,往前走一個,即可刪除本來排在第二個位置的元素現在排在了第一個位置上的元素。
更改后的核心代碼:
for(int i=0;i<list.size();i++){
System.out.println("即將刪除的元素:"+list.get(i));
list.remove(i);
i--;
}
結果:
即將刪除的元素:a
即將刪除的元素:b
即將刪除的元素:c
即將刪除的元素:d
即將刪除的元素:e
還剩余的元素個數:0
2、增強for循環
for(String x:list){ if(x.equals("del")) list.remove(x); }
這種方式的問題在於,刪除元素后繼續循環會報錯誤信息ConcurrentModificationException,因為元素在使用的時候發生了並發的修改,導致異常拋出。但是刪除完畢馬上使用break跳出,則不會觸發報錯。
原因:checkForComodification方法如果modCount不等於expectedModCount,則拋出ConcurrentModificationException異常。
具體原因及解決參考:https://www.cnblogs.com/dolphin0520/p/3933551.html#undefined
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錯誤。