Arrays.asList()方法的作用是將數組或一些元素轉為集合,而你得到的集合並不是我們通常使用的List集合,而是Arrays里面的一個內部類。阿里的開發手冊上java開發規范說到使用工具類Arrays.asList()方法把數組轉換成集合時,不能使用其修改集合相關的方法,它的add/remove/clear方法會拋出java.lang.UnsupportedOperationException的異常。
一、Arrays.asList的方法說明
public static void main(String[] args) { String[] strs = {"1","2","3"}; List<String> asList = Arrays.asList(strs); System.out.println(asList.size()); asList.add("4"); }
運行結果如下,asList的add和remove方法都會拋異常。
3 Exception in thread "main" java.lang.UnsupportedOperationException at java.util.AbstractList.add(AbstractList.java:148) at java.util.AbstractList.add(AbstractList.java:108) at com.basic.practice.ArraysPractice.main(ArraysPractice.java:12)
而set方法,則可以正常調用。
asList.set(1, "4");
方法沒有拋異常,正常返回結果:
3 [1, 4, 3] Process finished with exit code 0
查看下Arrays.asList()方法的源碼,可以發現asList里面直接用傳入的數組創建並返回了ArrayList,而這個ArrayList是Arrays的內部類,它的size()方法直接取的是構造方法傳參時的數組的長度,ArrayList雖然實現了List接口,但是並沒有重寫add和remove方法,但重寫了get和set方法。
@SafeVarargs @SuppressWarnings("varargs") public static <T> List<T> asList(T... a) { return new ArrayList<>(a); } /** * @serial include */ 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; } @Override public E get(int index) { return a[index]; } @Override public E set(int index, E element) { E oldValue = a[index]; a[index] = element; return oldValue; } @Override public int indexOf(Object o) { E[] a = this.a; if (o == null) { for (int i = 0; i < a.length; i++) if (a[i] == null) return i; } else { for (int i = 0; i < a.length; i++) if (o.equals(a[i])) return i; } return -1; } @Override public boolean contains(Object o) { return indexOf(o) != -1; } @Override public Spliterator<E> spliterator() { return Spliterators.spliterator(a, Spliterator.ORDERED); } @Override public void forEach(Consumer<? super E> action) { Objects.requireNonNull(action); for (E e : a) { action.accept(e); } } @Override public void replaceAll(UnaryOperator<E> operator) { Objects.requireNonNull(operator); E[] a = this.a; for (int i = 0; i < a.length; i++) { a[i] = operator.apply(a[i]); } } @Override public void sort(Comparator<? super E> c) { Arrays.sort(a, c); } }
org.springframework.util.CollectionUtils.arrayToList()方法內部上也是會有這樣的問題,底層也是使用的Arrays.asList()的方法。
二、常見的數組轉list的方式
1、java8的流轉換
List<String> list = Stream.of(strs).collect(Collectors.toList());
2、Collections工具類
ArrayList<String> list1 = new ArrayList<>(); Collections.addAll(list1,strs);
3、多一層封裝創建
List<String> strings = new ArrayList<>(Arrays.asList(strs));