- 通过列举方式传入primitive和引用类型时,每一个参数都作为List的一个元素
- 当传入数组时,如果是引用类型的数组,则数组每个元素都作为List的一个元素;如果是primitive类型的数组,问题就来了,这个数组会整个被当作List的一个元素
看下例:
Integer[] arrInteger = new Integer[10];
arrInteger[0] = 1;
arrInteger[1] = 2;
// 引用类型数组传入
System.out.println(Arrays.asList(arrInteger).get(0));
// 列举方式传入
System.out.println(Arrays.asList(1,2,3).get(0));
int[] arrInt = new int[10];
arrInt[0] = 1;
arrInt[1] = 2;
// int[] 数组传入:出问题
System.out.println(Arrays.asList(arrInt).get(0));
// int/char/short/double/boolean... 都是如此
输出:
1
1
[I@d542094
问题其实是出在java的可变参数类型上了: Arrays.asList(T… arr)
为什么可变参数类型传入primitive类型数组时就会变成一个单一的元素呢?
T… arr 的声明等效于T[] arr的声明,编译器会在caller传入的地方初始化出一个数组,将所有的变参加进数组,然后穿给callee。
范型只针对引用类型,范型T如果不通过extends指定父类型,默认就是Object类型,所以这个方法的字节码签名其实是:
([Ljava/lang/Object;)Ljava/util/List;
当primitive类型以列举的方式传入时,如
Arrays.asList(1, 2, 3)
由于这些int型参数不是引用类型,会触发编译器对每一个变参做autoboxing,生成Integer类型的数组,传入到asList
而当int类型数组传入时,数组已经是一个引用类型(类型为[I )了,autoboxing无需再起作用,直接就把数组作为一个Object传给asList方法了。
Java Autoboxing的触发时机
Converting a primitive value (an int, for example) into an object of the corresponding wrapper class (Integer) is called autoboxing. The Java compiler applies autoboxing when a primitive value is:
- Passed as a parameter to a method that expects an object of the corresponding wrapper class.
- Assigned to a variable of the corresponding wrapper class.
Auto unboxing的触发时机
Converting an object of a wrapper type (Integer) to its corresponding primitive (int) value is called unboxing. The Java compiler applies unboxing when an object of a wrapper class is:
- Passed as a parameter to a method that expects a value of the corresponding primitive type.
- Assigned to a variable of the corresponding primitive type.
https://docs.oracle.com/javase/tutorial/java/data/autoboxing.html
延伸知识:
primitive类型数组和引用类型数组的区别
任何引用类型的数组都可以复制给Object[] 类型变量,但是primitive类型数组就不行:
Object[] a = new Integer[2]; // 有warning,但编译通过
Object[] a = new int[2]; //编译出错