Arrays.asList(),我的数据去哪啦?

一、我的数据去哪了?

我们在项目中经常使用Arrays.asList(),它接收一个泛型可变参数,返回一个不可变List,我们看一下其方法结构:

    public static  List asList(T... a) {
        return new ArrayList<>(a);
    }

Arrays.asList这种方式要比原生的使用方式一个一个添加元素要简便很多,先看下示例:

      //使用asList的示例代码:
      List list = Arrays.asList(new String[]{"hello","world"});

      // 不使用asList的代码(要冗长很多,不太美观):
      List 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 list = Arrays.asList(arr);

我尝试将int[]转换成一个Integer类型的List存储,看起来没毛病.编译器却给出了一个错误提示:

Type mismatch: cannot convert from List to List

什么鬼?我传入了一个int数组,怎么给我转成了List,不是应该List么?由于基本类型没有泛型,那至少也给我个List啊。

应该是编译器误会了我的意思,我猜想。我传入一个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。和原来的想法简直是"不谋而合".说这话自己都不信~呵呵。

三、结论
  1. 使用Arrays.asList时不要使用基本数据类型的数组
  2. 基本数据类型不是协变的(int[] 不可转换成 Object[])
四、备注:

主要是一个技术群有人问到了这个问题,所以自己研究了一下,也顺便复习了一下泛型、擦除、自动装箱等等。如果有讲错的地方,麻烦给我指出。谢谢!

你可能感兴趣的:(Arrays.asList(),我的数据去哪啦?)