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