为什么Java不会将int []自动装箱为Integer []


问题内容

当我执行以下操作时,

  • arrayList1-包含一个元素,它是一个int[]
  • arrayList2-不编译(错误:构造函数ArrayList<Integer>(List<int[]>)未定义)
  • arrayList3-包含7个元素,它们是Integer对象

这是代码:

int[] intArray = new int[]{2,3,4,5,6,7,8};
ArrayList arrayList1 = new ArrayList(Arrays.asList(intArray));
ArrayList<Integer> arrayList2 = new ArrayList<Integer>(Arrays.asList(intArray));

Integer[] integerArray = new Integer[]{2,3,4,5,6,7,8};
ArrayList<Integer> arrayList3 = new ArrayList<Integer>(Arrays.asList(integerArray));

问题: 编译器为什么不自动将int[]to
中的元素装箱Integer并创建一个ArrayList<Integer>?这是什么原因呢?是我的愚蠢还是其他原因?


问题答案:

区别在于int[]本身是Object,而是Integer[]Integer对象的引用数组。

Arrays.asList(T...)方法采用某种类型的变量参数,T没有上限。该方法的擦除为Arrays.asList(Object...)。这意味着它将接受从扩展的任何类型的可变数量的参数Object

由于int不是Object,而是原始类型,因此不能将其作为的单个元素进行传递T[],而本身int[]Object本身,则它将作为T[]数组的第一个元素使用(T...内部是T[]唯一的)。但是,Integer[]将以形式传递T[],每个引用以形式Integer[]传递给T[]

即使您认为编译器应该已经完成​​了将int[]数组的每个元素转换Integer为的工作,对于编译器来说,这也将花费太多精力。首先,它将需要获取每个数组元素,Integer然后将其包装到,然后需要Integer[]从这些元素内部创建一个。真的太多了。它已经具有从int[]到的直接转换Object。尽管我一直希望Java允许从int[]到的隐式转换Integer[],这将使使用泛型时的工作变得更简单,但同样,这也是该语言的设计方式。

举一个简单的例子:

Object[] array = new Integer[10];  // this is valid conversion
Object[] array2 = new int[10];     // this is not
Object obj = new int[10];          // this is again a valid conversion

因此,在您的代码中Arrays.asList(intArray)返回a ArrayList<int[]>和not
ArrayList<Integer>。您不能将其传递给ArrayList<Integer>()构造函数。


有关:

  • int[]和Integer[]:有什么区别?