淺談Java list.remove( )方法需要注意的兩個坑
list.remove
最近做項目的過程中,需要用到list.remove()方法,結(jié)果發(fā)現(xiàn)兩個有趣的坑,經(jīng)過分析后找到原因,記錄一下跟大家分享一下。
代碼
直接上一段代碼,進行分析。
public class Main { public static void main(String[] args) { List<String> stringList = new ArrayList<>();//數(shù)據(jù)集合 List<Integer> integerList = new ArrayList<>();//存儲remove的位置 stringList.add("a"); stringList.add("b"); stringList.add("c"); stringList.add("d"); stringList.add("e"); integerList.add(2); integerList.add(4);//此處相當(dāng)于要移除最后一個數(shù)據(jù) for (Integer i :integerList){ stringList.remove(i); } for (String s :stringList){ System.out.println(s); } } }
如上代碼我們有一個5個元素的list數(shù)據(jù)集合,我們要刪除第2個和第4個位置的數(shù)據(jù)。
第一次運行
咦,為什么執(zhí)行兩次remove(),stringList的數(shù)據(jù)沒有變化呢?
沒有報錯,說明代碼沒有問題,那問題出在哪呢?
仔細分析我們發(fā)現(xiàn),remove()這個方法是一個重載方法,即remove(int position)和remove(object object),唯一的區(qū)別是參數(shù)類型。
for (Integer i :integerList){ stringList.remove(i); }
仔細觀察上面代碼你會發(fā)現(xiàn),其實i是Integer對象,而由于Java系統(tǒng)中如果找不到準(zhǔn)確的對象,會自動向上升級,而(int < Integer < Object),所以在調(diào)用stringList.remove(i)時,其實使用的remove(object object),而很明顯stringList不存在Integer對象,自然會移除失?。?.0),Java也不會因此報錯。
如果我們想使用remove(int position)方法,只能降低對象等級,即修改代碼;
for (Integer i :integerList){ int a =i; stringList.remove(a); }
第二次運行
我們發(fā)現(xiàn)提示在坐標(biāo)為4的地方越界了,這是為什么呢?
其實很簡單,因為執(zhí)行stringList.remove(2)后,list.size()就-1為4了,我們原來要移除的最后一個位置的數(shù)據(jù)移動到了第3個位置上,自然就造成了越界。
我們修改代碼先執(zhí)行stringList.remove(4),再執(zhí)行執(zhí)行stringList.remove(2)。
integerList.add(4);
integerList.add(2);
這個錯誤提醒我們:使用remove()的方法時,要先從大到小的位置移除。當(dāng)然如果你知道具體的對象,直接移除remove(對象)更穩(wěn)妥。
第三次執(zhí)行
嗯,這次沒問題了。
總結(jié)
1、使用remove()的方法時,要先從大到小的位置移除。當(dāng)然如果你知道具體的對象,直接移除remove(對象)更穩(wěn)妥。
2、要密切注意自己調(diào)用的remove()方法中的,傳入的是int類型還是一個對象。
補充知識: 關(guān)于List.remove()報錯的問題
我們?nèi)绻雱h掉List中某一個對象,我們可能就會想到會用List.remove()方法。但是這樣如果后續(xù)操作這個list的時候就會報錯。
具體的原因是當(dāng)你操作了List的remove方法的時候,他回去修改List的modCount屬性。
導(dǎo)致拋出異常java.util.ConcurrentModificationException。
最好的想要修改List對象,我們可以用ListIterator。
就像這樣:
ArrayList<Integer> arrayList = new ArrayList<>(); for (int i = 0; i < 20; i++) { arrayList.add(Integer.valueOf(i)); } ListIterator<Integer> iterator = arrayList.listIterator(); while (iterator.hasNext()) { if(需要滿足的條件){ iterator.remove();//刪除操作 iterator.add(integer);//新增操作 } }
這樣就不會去修改List的modCount屬性。
以上這篇淺談Java list.remove( )方法需要注意的兩個坑就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考,也希望大家多多支持腳本之家。
- Java contains用法示例
- 如何解決Mybatis--java.lang.IllegalArgumentException: Result Maps collection already contains value for X
- java Iterator.remove()實例方法分析
- Java中List遍歷刪除元素remove()的方法
- java 較大數(shù)據(jù)量取差集,list.removeAll性能優(yōu)化詳解
- java ArrayList.remove()的三種錯誤用法以及六種正確用法詳解
- JAVA的LIST接口的REMOVE重載方法調(diào)用原理解析
- Java中ArrayList在foreach里remove的問題詳析
- Java 集合的Contains和Remove方法
相關(guān)文章
SpringMVC?bean實現(xiàn)加載控制方法詳解
SpringMVC是一種基于Java,實現(xiàn)了Web?MVC設(shè)計模式,請求驅(qū)動類型的輕量級Web框架,即使用了MVC架構(gòu)模式的思想,將Web層進行職責(zé)解耦?;谡埱篁?qū)動指的就是使用請求-響應(yīng)模型,框架的目的就是幫助我們簡化開發(fā),SpringMVC也是要簡化我們?nèi)粘eb開發(fā)2022-08-08Java?empty、null、blank的區(qū)別小結(jié)
本文主要介紹了Java?empty、null、blank的區(qū)別小結(jié),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-06-06Java使用Request獲取請求參數(shù)的通用方式詳解
這篇文章主要給大家介紹了關(guān)于Java使用Request獲取請求參數(shù)的通用方式,在Java后端開發(fā)中第一步就是獲取前端傳過來的請求參數(shù),文中通過代碼介紹的非常詳細,需要的朋友可以參考下2024-01-01