Arrays.asList()你真的知道怎么用吗?

发现问题

前几天在看别人的项目的时候,发现一个问题,简单复现一下这个问题

// 注意这是一个Integer对象的数组哦
Integer[] arr = new Integer[]{9999,88,77};
List<Integer> list = Arrays.asList(arr);
// 执行以下操作,有问题么?
list.add(1);
list.remove(0);

好的,如果你觉得没错,和我刚开始的想法一致。在我没有认真学习这个asList方法时,天真的以为没有问题,顾名思义啊,就是把数组转换成List呗。

// 恭喜,喜提报错,如果是这样测试该找你麻烦了,/(ㄒoㄒ)/~~ 
Exception in thread "main" java.lang.UnsupportedOperationException
	at java.util.AbstractList.add(AbstractList.java:148)
	at java.util.AbstractList.add(AbstractList.java:108)
	at TestForAsList.main(TestForAsList.java:13)

废话不多说,我们直接进源码里面看看。

@SafeVarargs
@SuppressWarnings("varargs")
public static <T> List<T> asList(T... a) {
    return new ArrayList<>(a);  // 看起来没有任何异常,emm。然而呢,这个ArrayList并不是Java.util包下的ArrayList,而是一个在Arrays下实现的内部类。
}

image

我们发现这个静态内部类里面并没有实现List的add和remove方法。那么子类将延用父类AbstractList的方法实现,这个继承应没有什么问题。

// 我们进入这个父类的实现,发现了报错的根源。。
public abstract class AbstractList<E> extends AbstractCollection<E> implements List<E> {
    
	public void add(int index, E element) {
        throw new UnsupportedOperationException();
    }


    public E remove(int index) {
        throw new UnsupportedOperationException();
    }
}

总结:

究其原因,就是这个asList方法返回的是一个内部类,只实现了一些遍历以及更新的方法。下次使用它的时候需要注意一下。

扩展知识点

数组值的变化

然后我就发现了这个很有意思的点,就是这个Arrays包下的ArrayList用的是构造器传进来的数组,并不像我们原来认为的ArrayList的那样会拷贝数组然后创建一个新的数组。

这意味着我们在对这个List进行set的改动时,我们同时会更改原数组的值

public static void main(String[] args) {
        Integer[] arr = new Integer[]{9999,88,77};
        List<Integer> list = Arrays.asList(arr);
        
        list.set(1,0);
        System.out.println(list.get(1));  // 输出0
        System.out.println(arr[1]);  // 同样也输出了0
}

int[]数组

我们都知道int是基本数据类型,如果我们向asList()方法里面传入一个int[]数组会发生什么呢?

我们知道泛型的话需要的是对象类型,基础数据类型是不能作为泛型的。

// 此时int[]会作为一个对象类型,然后转换为list。此时长度为1,且可以正常的取出来作为list元素
public static void main(String[] args) {
        int[] arr = new int[]{9999,88,77};
        List<int[]> list = Arrays.asList(arr);
        System.out.println(list.size());
        System.out.println(list.get(0)[0]);
        System.out.println(list.get(0)[1]);
        System.out.println(list.get(0)[2]);
}

好了,asList()方法就学废了。一天学废一个小知识/dog

posted @ 2022-09-05 23:47  大队长11  阅读(750)  评论(1编辑  收藏  举报