您好,登錄后才能下訂單哦!
今天就跟大家聊聊有關(guān)深入淺析Java 中 List 集合類的一些問題,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。
第一個坑:Arrays.asList 方法返回的 List 不支持增加、刪除操作
例如我們執(zhí)行以下代碼:
List<String> strings = Arrays.asList("m", "g"); strings.add("h");
會拋出 java.lang.UnsupportedOperationException
異常,此時你內(nèi)心 OS what?明明返回的 ArrayList 為啥不能往里面增加元素,這以后還能好好的增加元素嗎?
,然后果斷開啟 Debug
大法:
發(fā)現(xiàn)返回的 ArrayList
并不是我們常用的 java.util.ArrayList
,而是 Arrays
的內(nèi)部類 java.util.Arrays.ArrayList
。進入方法 Arrays.asList
源碼如下:
public static <T> List<T> asList(T... a) { return new ArrayList<>(a); }
方法返回的是 Arrays
的靜態(tài)內(nèi)部類 java.util.Arrays.ArrayList
,該類雖然和 java.util.ArrayList
也繼承自抽象類 java.util.AbstractList
,但是通過該類的源碼發(fā)現(xiàn)它并沒有對抽象父類 AbstractList
的 add
方法默認(rèn)就是拋出 java.lang.UnsupportedOperationException
異常。
這個坑的根本原因是我們調(diào)用返回的 strings
的 add
方法是繼承自抽象父類的 add
方法,而抽象父類的方法默認(rèn)就是拋出 java.lang.UnsupportedOperationException
這個異常。
第二個坑,Arrays.asList 方法返回的新 List 和該方法原始入?yún)?shù)組修改會相互影響
Arrays.asList
方法除了上面這個 不支持增加、刪除元素 這個坑之外,還有另外一個坑:
從以上代碼可以發(fā)現(xiàn),對原始數(shù)組的修改會影響我們通過 Arrays.asList
方法獲得的新 List
,深入 java.util.Arrays.ArrayList
的源碼:
private static class ArrayList<E> extends AbstractList<E> implements RandomAccess, java.io.Serializable { private static final long serialVersionUID = -2764017481108945198L; private final E[] a; ArrayList(E[] array) { a = Objects.requireNonNull(array); } ... }
可以發(fā)現(xiàn)是直接使用了原始的數(shù)組,所有當(dāng)我們使用 Arrays.asList
方式獲得的 List
時要特別注意,因為共享了數(shù)組,相互修改時可能產(chǎn)生一些意想不到的 Bug。標(biāo)準(zhǔn)的姿勢之一是將其作為 ArrayList
構(gòu)造方法的參數(shù)重新 new
一個 List
出來即可(e.g. List<String> stringList = new ArrayList<>(Arrays.asList(arrays))
)或者通過 Guava
庫中的 Lists.newArrayList
,將返回的新 List
和原始的數(shù)組解耦,就不會再互相影響了。
第三個坑,直接遍歷 List 集合刪除元素會報錯
在直接遍歷集合元素時增加、刪除元素會報錯,比如執(zhí)行如下代碼:
List<String> stringList = Lists.newArrayList("m", "g", "h"); for (String s : stringList) { if (Arrays.asList("m", "h").contains(s)) { stringList.remove(s); } }
以上代碼可以正常編譯通過,但是執(zhí)行時會拋出 java.util.ConcurrentModificationException
異常,查看其源碼可以發(fā)現(xiàn),刪除元素方法 remove
會使集合結(jié)構(gòu)發(fā)生修改,也就是 modCount(
集合實際修改的次數(shù))會修改,在循環(huán)過程中,會比較當(dāng)前 List
的集合實際修改的次數(shù) modCount
與迭代器修改的次數(shù) expectedModCount
,而 expectedModCount
是初始化時的 modCount
, 二者不相等,就會報 ConcurrentModificationException
異常。解決方法主要有兩種方式,1.使用 ArrayList
的迭代器方式遍歷,然后調(diào)用其中的方法。2.在 JDK 1.8+ 可以使用 removeIf
方法進行刪除操作。
最后扎心一問:調(diào)用 ArrayList
的 remove
方法傳入 int
基本類型的數(shù)字和 Integer
包裝類型的數(shù)字,執(zhí)行結(jié)果是不是一樣的?
看完上述內(nèi)容,你們對深入淺析Java 中 List 集合類的一些問題有進一步的了解嗎?如果還想了解更多知識或者相關(guān)內(nèi)容,請關(guān)注億速云行業(yè)資訊頻道,感謝大家的支持。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。