Arrays.asList存在的坑

我想問問天發表於2019-06-24

引語:

阿里巴巴java開發規範說到使用工具類Arrays.asList()方法把陣列轉換成集合時,不能使用其修改集合相關的方法,它的add/remove/clear方法會丟擲UnsupportedOperationException(),我們來看一下為什麼會出現這種情況.

問題分析:

我們做個測試

public static void main(String[] args) {
       List<String> list = Arrays.asList("a", "b", "c");
       // list.clear();
       // list.remove("a");
       // list.add("g");
   }
複製程式碼

被註釋的三行可以分別解開註釋,執行後確實出現了規約中所說的異常.我們來看下Arrays.asList()做了什麼操作.

public static <T> List<T> asList(T... a) {
       return new ArrayList<>(a);
   }
複製程式碼

看上去是個很正常的方法,然而實際上你點進到ArrayList發現,其實ArrayList並不是我們平時用的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);
       }

       @Override
       public int size() {
           return a.length;
       }

       @Override
       public Object[] toArray() {
           return a.clone();
       }

       @Override
       @SuppressWarnings("unchecked")
       public <T> T[] toArray(T[] a) {
           int size = size();
           if (a.length < size)
               return Arrays.copyOf(this.a, size,
                                    (Class<? extends T[]>) a.getClass());
           System.arraycopy(this.a, 0, a, 0, size);
           if (a.length > size)
               a[size] = null;
           return a;
       }
       // 後面省略了
複製程式碼

而是Arrays裡面的一個內部類.而且這個內部類沒有add,clear,remove方法,所以丟擲的異常其實來自於AbstractList.

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

   public E remove(int index) {
      throw new UnsupportedOperationException();
  }
複製程式碼

點進去就會發現丟擲異常的地方,clear底層也會呼叫到remove所以也會丟擲異常.

總結:

1.Arrays.asList()不要亂用,底層其實還是陣列;
2.如果使用了Arrays.asList()的話,最好不要使用其集合的操作方法;
3. List list = new ArrayList<>(Arrays.asList("a", "b", "c"))可以在外面這樣包一層真正的ArrayList(陣列轉集合有很多方式,可以參考連結).

相關文章