Arrays.asList注意事项
Arrays.asList()将数组转换为集合后,底层其实还是数组,《阿里巴巴Java 开发手册》对于这个方法有如下描述:
一、基本类型数组传入转化问题
int[] datas = new int[]{1,2,3,4,5}; List list = Arrays.asList(datas); System.out.println(list.size());
运行结果是1。一个长度为5的数组,转化为List后,长度却成了1。
Integer[] datas = new Integer[]{1,2,3,4,5}; List list = Arrays.asList(datas); System.out.println(list.size());
public static <T> List<T> asList(T... a) { return new ArrayList<>(a); }
asList接受的是一个泛型类型的参数,再构造了一个ArrayList。然而基本类型是不支持泛型化的,但是数组支持,所以采用基本类型的数组转化后是将数组放入了构造的ArrayList中,长度是1。
二、使用Arrays.asList()转换后的集合不能进行增删操作
阿里巴巴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所以也会抛出异常。
通过Arrays.asList方法转换的List并不是真正的java.util.ArrayList!!!返回的是java.util.Arrays.ArrayList!!!
三、正确的使用
支持基础类型的方式
- 如果使用Spring
int[] a = {1,2,3}; List list = CollectionUtils.arrayToList(a); System.out.println(list);
- 如果使用Java 8(推荐)
int intArray[] = {1, 2, 3}; List<Integer> iList = Arrays.stream(intArray) .boxed() .collect(Collectors.toList()); System.out.println(iList);
数组转ArrayList
- 遍历转换
Integer intArray[] = {1, 2, 3}; ArrayList<Integer> aList = new ArrayList<>(); for (Integer i: intArray){ aList.add(i); }
然这种方式不够优雅!反正我不愿意使用。
- 使用工具类
上面方案不够优雅,那么这种相对来说优雅一些。
List<String> list = new ArrayList(); Collections.addAll(list, "welcome", "to", "china");
你以为这种还不错?
too young too simple!
addAll()方法的实现就是用的上面遍历的方式。
- 如果使用Java 8(推荐)
既可以用于基本类型也可以返回想要的集合。
int intArray[] = {1, 2, 3}; List<Integer> iList = Arrays.stream(intArray) .boxed() .collect(Collectors.toList()); System.out.println(iList);
- 两个集合类结合
ArrayList arrayList = new ArrayList<>(Arrays.asList("welcome", "to", "china"));