Java如何在List或Map遍歷過程中刪除元素
遍歷刪除List或Map中的元素有很多種方法,當(dāng)運(yùn)用不當(dāng)?shù)臅r候就會產(chǎn)生問題。下面通過這篇文章來再學(xué)習(xí)學(xué)習(xí)吧。
一、List遍歷過程中刪除元素
使用索引下標(biāo)遍歷的方式
示例:刪除列表中的2
public static void main(String[] args) { List<Integer> list = new ArrayList<Integer>(); list.add(1); list.add(2); list.add(2); list.add(3); list.add(4); for (int i = 0; i < list.size(); i++) { if(2 == list.get(i)){ list.remove(i); } System.out.println(list.get(i)); } System.out.println("list=" + list.toString()); }
輸出結(jié)果:
1 2 3 4 list=[1, 2, 3, 4]
問題:
結(jié)果顯示只刪除了一個2,另一個2被遺漏了,原因是:刪除了第一個2后,集合里的元素個數(shù)減1,后面的元素往前移了1位,導(dǎo)致了第二個2被遺漏了。
使用For循環(huán)遍歷的方式
示例:
public static void listIterator2(){ List<Integer> list = new ArrayList<Integer>(); list.add(1); list.add(2); list.add(2); list.add(3); list.add(4); for (int value : list) { if(2 == value){ list.remove(value); } System.out.println(value); } System.out.println("list=" + list.toString()); }
結(jié)果:
Exception in thread "main" 1 2 java.util.ConcurrentModificationException at java.util.ArrayList$Itr.checkForComodification(Unknown Source) at java.util.ArrayList$Itr.next(Unknown Source) at test.ListIterator.listIterator2(ListIterator.java:39) at test.ListIterator.main(ListIterator.java:10)
說明:
jdk中對ConcurrentModificationException的描述:
public class ConcurrentModificationException extends
RuntimeException當(dāng)方法檢測到對象的并發(fā)修改,但不允許這種修改時,拋出此異常。
例如,某個線程在 Collection 上進(jìn)行迭代時,通常不允許另一個線性修改該 Collection。通常在這些情況下,迭代的結(jié)果是不確定的。如果檢測到這種行為,一些迭代器實(shí)現(xiàn)(包括 JRE 提供的所有通用 collection 實(shí)現(xiàn))可能選擇拋出此異常。執(zhí)行該操作的迭代器稱為快速失敗 迭代器,因為迭代器很快就完全失敗,而不會冒著在將來某個時間任意發(fā)生不確定行為的風(fēng)險。
注意:此異常不會始終指出對象已經(jīng)由不同 線程并發(fā)修改。如果單線程發(fā)出違反對象協(xié)定的方法調(diào)用序列,則該對象可能拋出此異常。例如,如果線程使用快速失敗迭代器在 collection 上迭代時直接修改該 collection,則迭代器將拋出此異常。
注意:迭代器的快速失敗行為無法得到保證,因為一般來說,不可能對是否出現(xiàn)不同步并發(fā)修改做出任何硬性保證??焖偈〔僮鲿M最大努力拋出 ConcurrentModificationException
。因此,為提高此類操作的正確性而編寫一個依賴于此異常的程序是錯誤的做法,正確做法是:ConcurrentModificationException
應(yīng)該僅用于檢測 bug。
Java中的For each實(shí)際上使用的是iterator進(jìn)行處理的。而iterator是不允許集合在iterator使用期間刪除的。所以導(dǎo)致了iterator拋出了ConcurrentModificationException
。
正確的方式
示例:
public static void listIterator3(){ List<Integer> list = new ArrayList<Integer>(); list.add(1); list.add(2); list.add(2); list.add(3); list.add(4); Iterator<Integer> it = list.iterator(); while (it.hasNext()){ Integer value = it.next(); if (2 == value) { it.remove(); } System.out.println(value); } System.out.println("list=" + list.toString()); }
結(jié)果:
1 2 2 3 4 list=[1, 3, 4]
二、Map遍歷過程中刪除元素
正確做法的示例:
public static void main(String[] args) { HashMap<String, String> map = new HashMap<String, String>(); map.put("1", "test1"); map.put("2", "test2"); map.put("3", "test3"); map.put("4", "test4"); //完整遍歷Map for (Entry<String, String> entry : map.entrySet()) { System.out.printf("key: %s value:%s\r\n", entry.getKey(), entry.getValue()); } //刪除元素 Iterator<Map.Entry<String, String>> it = map.entrySet().iterator(); while(it.hasNext()) { Map.Entry<String, String> entry= it.next(); String key= entry.getKey(); int k = Integer.parseInt(key); if(k%2==1) { System.out.printf("delete key:%s value:%s\r\n", key, entry.getValue()); it.remove(); } } //完整遍歷Map for (Entry<String, String> entry : map.entrySet()) { System.out.printf("key: %s value:%s\r\n", entry.getKey(), entry.getValue()); } }
結(jié)果:
key: 1 value:test1 key: 2 value:test2 key: 3 value:test3 key: 4 value:test4 delete key:1 value:test1 delete key:3 value:test3 key: 2 value:test2 key: 4 value:test4
注意
但對于iterator的remove()
方法,也有需要我們注意的地方:
每調(diào)用一次iterator.next()
方法,只能調(diào)用一次remove()
方法。
調(diào)用remove()
方法前,必須調(diào)用過一次next()
方法。
JDK-API中對于remove()方法的描述:
void remove()
從迭代器指向的集合中移除迭代器返回的最后一個元素(可選操作)。每次調(diào)用 next 只能調(diào)用一次此方法。如果進(jìn)行迭代時用調(diào)用此方法之外的其他方式修改了該迭代器所指向的集合,則迭代器的行為是不明確的。
拋出:UnsupportedOperationException
- 如果迭代器不支持 remove
操作。IllegalStateException
- 如果尚未調(diào)用 next
方法,或者在上一次調(diào)用 next
方法之后已經(jīng)調(diào)用了remove
方法。
總結(jié)
以上就是關(guān)于List與Map的遍歷過程中刪除元素的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作能帶來一定的幫助,如果有疑問大家可以留言交流。
相關(guān)文章
Spring注解驅(qū)動之ApplicationListener用法解讀
這篇文章主要介紹了Spring注解驅(qū)動之ApplicationListener用法解讀,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-09-09Java算法實(shí)現(xiàn)調(diào)整數(shù)組順序使奇數(shù)位于偶數(shù)之前的講解
今天小編就為大家分享一篇關(guān)于Java算法實(shí)現(xiàn)調(diào)整數(shù)組順序使奇數(shù)位于偶數(shù)之前的講解,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2019-01-01Spring Cloud Gateway內(nèi)置的斷言和過濾器作用說明
這篇文章主要介紹了Spring Cloud Gateway內(nèi)置的斷言和過濾器作用說明,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-06-06Java實(shí)現(xiàn)醫(yī)院管理系統(tǒng)
這篇文章主要介為大家詳細(xì)紹了Java實(shí)現(xiàn)醫(yī)院管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-12-12