Java踩坑記系列之Arrays.AsList

Java老K發表於2020-11-01

java.util.ArraysasList方法可以方便的將陣列轉化為集合,我們平時開發在初始化ArrayList時使用的比較多,可以簡化程式碼,但這個靜態方法asList()有幾個坑需要注意:

一. 如果對集合使用增加或刪除元素的操作將會報錯

如下程式碼:

List list = Arrays.asList("a","b","c");
list.add("d");

輸出結果:

Exception in thread "main" java.lang.UnsupportedOperationException
  at java.util.AbstractList.add(AbstractList.java:148)
  at java.util.AbstractList.add(AbstractList.java:108)
  at ArraysAsListTest.main(ArraysAsListTest.java:7)

Arrays.asList的原始碼雖然返回的是一個ArrayList,但這個ArrayListArrays內部的一個靜態類(java.util.Arrays.ArrayList),並不是我們平時用的java.util.ArrayList只是名字一樣,包名不一樣:

file

這個ArrayList也繼承自AbstractList,但是卻沒有實現add()remove()方法用這兩個方法其實呼叫的是AbstractList的預設方法:

public void add(int index, E element) {
    throw new UnsupportedOperationException();
}

直接丟擲" UnsupportedOperationException"的異常!

所以當我們呼叫這個ArrayListadd方法時其實是呼叫了抽象集合類裡的add方法,就丟擲了上面的異常,原因是陣列雖然提供了轉集合的方法, 但本質上還是個陣列,陣列的長度是固定的,所以jdk的作者沒有實現add()remove()方法。

大家在使用的時候需要注意,要結合具體業務場景判斷:如果只是對轉成後的集合進行遍歷或使用stream()流操作都沒有問題,可以正常使用(修改元素也沒問題) 但如果需要新增元素或刪除元素等改變集合長度的操作就要避免,否則報錯就會影響到我們正常的業務邏輯。

二. 長度問題

還是ArraysasList(),該方法無法將一個基本型別的陣列轉換成集合List,或者說基本型別雖然能轉換成功,但不是"我們想要的集合",因為asList方法接收的是一個泛型的變長引數。

而基本資料型別是無法被泛型化的,這樣的話對於泛型而言,基本資料型別會被作為一個"[x"的型別,"["表示這是一個陣列,"x"為當前陣列的型別,例如: int[] 實際上它的型別是"[I"

file

這樣的話Arrays.asList傳入的引數是基本型別的話,會把陣列當做一個元素處理(長度=1),如果我們要遍歷List或獲取List的長度就不準確了,類似下面的用法:

int array[] = {1,2,3,4,5};
List list = Arrays.asList(array);
System.out.println("長度="+list.size());
System.out.println("是否含有元素3="+list.contains(3));

輸出結果:

長度=1
是否含有元素3=false

如果我們的程式碼裡用到了這樣獲取長度或判斷是否包含指定元素的操作就要注意寫法,否則就會影響我們的邏輯流程!

為避免這樣的問題出現,有很多種其他寫法,比如將array轉成包裝類(即引用型別),如下:

Integer array[] = {1,2,3,4,5};
List list = Arrays.asList(array);
System.out.println("長度="+list.size());
System.out.println("是否含有元素3="+list.contains(3));

輸出結果:

長度=5
是否含有元素3=true

當然也可以使用google的集合類"Ints.asList",如下:

int array[] = {1,2,3,4,5};
List list = Ints.asList(array);
System.out.println("長度="+list.size());
System.out.println("是否含有元素3="+list.contains(3));

輸出結果:

長度=5
是否含有元素3=true

但是google的Ints.asList方法返回的list也是一個自己實現的集合,同樣不支援addremove方法,如果既要保證轉換成list後的資料長度正確又要能增刪操作的話,就用java.util.ArrayList吧。

三. Arrays.asList方法返回的是陣列的一個檢視,也就是說對這個list的操作都會反映在原陣列上

文章來源:http://javakk.com/132.html

本文由部落格群發一文多發等運營工具平臺 OpenWrite 釋出