<导航

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());
运行结果为5。
同样的代码,只是将数组的类型由原来的基本类型改变为包装类型,转化后的List却不一样,这是为什么?
看一下asList的代码:
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;
 }
 // 后面省略了
View Code

而是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);
  • 两个集合类结合
 将Arrays.asList返回的集合作为ArrayList的构造参数
ArrayList arrayList = new ArrayList<>(Arrays.asList("welcome", "to", "china"));

 

 
 参考文章:
 
posted @ 2019-08-01 16:11  字节悦动  阅读(378)  评论(0编辑  收藏  举报