一、我的数据去哪了?
我们在项目中经常使用Arrays.asList(),它接收一个泛型可变参数,返回一个不可变List,我们看一下其方法结构:
public static <T> List<T> asList(T... a) {
return new ArrayList<>(a);
}
Arrays.asList这种方式要比原生的使用方式一个一个添加元素要简便很多,先看下示例:
//使用asList的示例代码:
List<String> list = Arrays.asList(new String[]{"hello","world"});
// 不使用asList的代码(要冗长很多,不太美观):
List<String> list = Lists.newArrayList();
list.add("hello");
list.add("world);
Arrays.asList()东西虽好,但是也不可忽视其隐含的一些问题:
下面先看一下程序:
int[] arr = {1,3,5};//基本类型int数组
System.out.println(Arrays.asList(arr).size());
我相信大部分人都能答上来,毕竟都上过小学~: 答案肯定是3
哈哈~开个玩笑,我觉得有一部分人应该上当了。
右键运行一下,结果发现竟然是1 ,简直不可想象。下面我详细分析一下原因。
二、错误原因分析
先贴断代码压压惊:
int[] arr = {1,3,5};
List<Integer> list = Arrays.asList(arr);
我尝试将int[]转换成一个Integer类型的List存储,看起来没毛病.编译器却给出了一个错误提示:
Type mismatch: cannot convert from List<int[]> to List<Integer>
什么鬼?我传入了一个int数组,怎么给我转成了List<int[]>,不是应该List<int>么?由于基本类型没有泛型,那至少也给我个List<Integer>啊。
应该是编译器误会了我的意思,我猜想。我传入一个int数组,编译器却把它当成一个int[]类型的对象传入进来,也就是Object[0] = arr;回到开头看asList方法结构,咦?接收一个泛型可变参数?我们都知道泛型是通过擦除来实现的,而泛型只是强化了传入的参数类型,擦除的目的只是为了兼容之前的老代码而已,一切都很正常。
可变参数其实就是一个数组参数,但是它比数组更智能,长度可以动态变化。结合着对泛型、擦除的理解,就可以将方法赋值进行如下转换:
int[] arr = {1,3,5};
Object[] param = arr;
可恶的编译器又给我报了一个错。说int[]不能转换成Object[],简直是啃爹。
数组本来就是协变的,既然int可以转换成Object,那为啥int[]就不能转换成Object[].真是奇怪~
javap看了一下字节码,发现Object obj = 1;这个执行语句会自动装箱,也就是Integer.valueOf(1);难道问题是arr为基本类型数组导致的?赶紧试一把:
Integer[] arr = {1,3,5};
System.out.println(Arrays.asList(arr).size());//3
返回是3。和原来的想法简直是"不谋而合".说这话自己都不信~呵呵。
三、结论
- 使用Arrays.asList时不要使用基本数据类型的数组
- 基本数据类型不是协变的(int[] 不可转换成 Object[])
四、备注:
主要是一个技术群有人问到了这个问题,所以自己研究了一下,也顺便复习了一下泛型、擦除、自动装箱等等。如果有讲错的地方,麻烦给我指出。谢谢!