方法用于返回仅包含指定对象的不可变列表,被限定只被分配一个内存空间,也就是只能存放一个元素的内容,因此又被称作为单例列表。见图5
优点:不会造成内存的浪费。
缺点:返回的集合不允许操作。
set(),add()不允许操作。执行方法报错:java.lang.UnsupportedOperationException
看下源码为什么不支持set()和add()操作呢?
首先图1我们可以看到singletonList(T O)的返回值为SingletonList<>(o)
图2我们可以看到SingletonList类继承了AbstractList
图3为父类AbstractList的add()方法,可以看到是直接返回了UnsupportedOperationException异常信息的。
图1
public static List 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
我们可以看到接口的参数是带有泛型的,那么我们看下以下几种情况。
执行以下代码段,我们可以发现执行的结果分别为:1,3,3
再看源码,asList接收的是一个泛型变长参数,而我们知道基本类型是不能泛型化的,就是说8种基本类型不能作为泛型参数,要想作为泛型参数就要使用其所对应的包装类。
但是listA的Size为什么是1呢,这是因为listA传递的是一个int类型的数组,数组是一个对象,它是可以泛型化的,也就是说例子中是把一个int类型的数组作为了T的类型,所以转换后在List中就只有一个类型为int数组的元素。后边ListA1与ListB也就可以理解了,一个是进行了自动打包,一个是本来就是包装类型。
public static List asList(T... a) {
return new ArrayList<>(a);
}
@org.junit.jupiter.api.Test
void asList() throws ParseException {
// List a = Arrays.asList("a");
// a.add("1");
// ArrayList 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 a = Arrays.asList("a");
// a.add("1");
// ArrayList 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 List 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