1:Collections.singletonList(T O)

方法用于返回仅包含指定对象的不可变列表,被限定只被分配一个内存空间,也就是只能存放一个元素的内容,因此又被称作为单例列表。见图5

优点:不会造成内存的浪费。

缺点:返回的集合不允许操作。

set(),add()不允许操作。执行方法报错:java.lang.UnsupportedOperationException

看下源码为什么不支持set()和add()操作呢?

首先图1我们可以看到singletonList(T O)的返回值为SingletonList<>(o)

图2我们可以看到SingletonList类继承了AbstractList

图3为父类AbstractList的add()方法,可以看到是直接返回了UnsupportedOperationException异常信息的。

图1

public static <T> List<T> singletonList(T o) {
        return new SingletonList<>(o);
    }

图2

 图3

 public boolean add(E e) {
        add(size(), e);
        return true;
    }

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

图4

图5

 2:Arrays.asList(T O)

我们可以看到接口的参数是带有泛型的,那么我们看下以下几种情况。

执行以下代码段,我们可以发现执行的结果分别为:1,3,3

再看源码,asList接收的是一个泛型变长参数,而我们知道基本类型是不能泛型化的,就是说8种基本类型不能作为泛型参数,要想作为泛型参数就要使用其所对应的包装类。

但是listA的Size为什么是1呢,这是因为listA传递的是一个int类型的数组,数组是一个对象,它是可以泛型化的,也就是说例子中是把一个int类型的数组作为了T的类型,所以转换后在List中就只有一个类型为int数组的元素。后边ListA1与ListB也就可以理解了,一个是进行了自动打包,一个是本来就是包装类型。
 

 public static <T> List<T> asList(T... a) {
        return new ArrayList<>(a);
    }
@org.junit.jupiter.api.Test
    void asList() throws ParseException {
//        List<String> a = Arrays.asList("a");
//        a.add("1");
//        ArrayList<String> list = new ArrayList<>();

        int[] a = {1,2,3};
        Integer[] b = {1,2,3};

        List listA = Arrays.asList(a);
        List listA1 = Arrays.asList(1,2,3);
        List listB = Arrays.asList(b);

        System.out.println(listA.size());//out:1
        System.out.println(listA1.size());//out:3
        System.out.println(listB.size());//out:3


    }

图6

 为了验证,我们再把每个集合元素的类型打印出来

@org.junit.jupiter.api.Test
    void asList() throws ParseException {
//        List<String> a = Arrays.asList("a");
//        a.add("1");
//        ArrayList<String> list = new ArrayList<>();

        int[] a = {1,2,3};
        Integer[] b = {1,2,3};

        List listA = Arrays.asList(a);
        List listA1 = Arrays.asList(1,2,3);
        List listB = Arrays.asList(b);
//
//        System.out.println(listA.size());//out:1
//        System.out.println(listA1.size());//out:3
//        System.out.println(listB.size());//out:3

        System.out.println("outA:" + listA.get(0).getClass());//out:1
        System.out.println("outA1:" +listA1.get(0).getClass());//out:3
        System.out.println("outB:" +listB.get(0).getClass());//out:3


    }

图7

 Arrays.asList的某些特性是和Collections.singletonList是相同的,对于add()方法和set()方法也是会抛出UnsupportedOperationException

看下源码:

 @SafeVarargs
    @SuppressWarnings("varargs")
    public static <T> List<T> asList(T... a) {
        return new ArrayList<>(a);
    }

Arrays.asList最终的返回结果是ArrayList。这时候是不是存在疑问,为什么ArrayList不支持add()和set()方法呢?

仔细观察图8和图9我们可以发现,虽然名字一样,但是两个ArrayList并不在同一个包下。

java.util.Arrays.ArrayList是Arrays的内部类,他的add方法继承了父类AbstractList的。

图8

 图9

Logo

为开发者提供学习成长、分享交流、生态实践、资源工具等服务,帮助开发者快速成长。

更多推荐