首页 » 编写高质量代码:改善Java程序的151个建议 » 编写高质量代码:改善Java程序的151个建议全文在线阅读

《编写高质量代码:改善Java程序的151个建议》建议65:避开基本类型数组转换列表陷阱

关灯直达底部

我们在开发过程中经常会使用Arrays和Collections这两个工具类在数组和列表之间转换,非常方便,但也有时候会出现一些奇怪的问题,来看如下代码:


public static void main(Stringargs){

intdata={1,2,3,4,5};

List list=Arrays.asList(data);

System.out.println("列表中的元素数量是:"+list.size());

}


也许你会说,这很简单,list变量的元素数量当然是5了。但是运行后打印出来的列表数量却是1。

事实上data确实是一个有5个元素的int类型数组,只是通过asList转换成列表后就只有1个元素了,这是为什么呢?其他4个元素到什么地方去了呢?

我们仔细看一下Arrays.asList的方法说明:输入一个变长参数,返回一个固定长度的列表。注意这里是一个变长参数,看源代码:


public static<T>List<T>asList(T……a){

return new ArrayList<T>(a);

}


asList方法输入的是一个泛型变长参数,我们知道基本类型是不能泛型化的,也就是说8个基本类型不能作为泛型参数,要想作为泛型参数就必须使用其所对应的包装类型。那前面的例子传递了一个int类型的数组,为什么程序没有报编译错呢?

在Java中,数组是一个对象,它是可以泛型化的,也就是说我们的例子是把一个int类型的数组作为了T的类型,所以转换后在List中就只有一个类型为int数组的元素了,我们打印出来看看,代码如下:


public static void main(String args){

intdata={1,2,3,4,5};

List list=Arrays.asList(data);

System.out.println("元素类型:"+list.get(0).getClass());

System.out.println("前后是否相等:"+data.equals(list.get(0)));

}


输出的结果是:


元素类型:class [I

前后是否相等:true


很明显,放在列表中的元素是一个int数组,可能有读者要问了,为什么“元素类型:”后的class是"[I"?我们并没有指明是数组(Array)类型呀!这是因为JVM不可能输出Array类型,因为Array是属于java.lang.reflect包的,它是通过反射访问数组元素的工具类。在Java中任何一个数组的类都是"[I",究其原因就是Java并没有定义数组这一个类,它是在编译器编译的时候生成的,是一个特殊的类,在JDK的帮助中也没有任何数组类的信息。

弄清楚了问题,修改方案也就诞生了,直接使用包装类即可,代码如下:


public static void main(String args){

Integerdata={1,2,3,4,5};

List list=Arrays.asList(data);

System.out.println("列表中的元素数量是:"+list.size());

}


仅仅修改了加粗字体部分,把int替换为Integer即可让输出元素数量为5。需要说明的是,不仅仅是int类型的数组有这个问题,其他7个基本类型的数组也存在相似的问题,这就需要读者注意了,在把基本类型数组转换成列表时,要特别小心asList方法的陷阱,避免出现程序逻辑混乱的情况。

注意 原始类型数组不能作为asList的输入参数,否则会引起程序逻辑混乱。