1700443521
编写高质量代码:改善Java程序的151个建议 建议65:避开基本类型数组转换列表陷阱
1700443522
1700443523
我们在开发过程中经常会使用Arrays和Collections这两个工具类在数组和列表之间转换,非常方便,但也有时候会出现一些奇怪的问题,来看如下代码:
1700443524
1700443525
public static void main(String[]args){
1700443526
1700443527
int[]data={1,2,3,4,5};
1700443528
1700443529
List list=Arrays.asList(data);
1700443530
1700443531
System.out.println(“列表中的元素数量是:”+list.size());
1700443532
1700443533
}
1700443534
1700443535
也许你会说,这很简单,list变量的元素数量当然是5了。但是运行后打印出来的列表数量却是1。
1700443536
1700443537
事实上data确实是一个有5个元素的int类型数组,只是通过asList转换成列表后就只有1个元素了,这是为什么呢?其他4个元素到什么地方去了呢?
1700443538
1700443539
我们仔细看一下Arrays.asList的方法说明:输入一个变长参数,返回一个固定长度的列表。注意这里是一个变长参数,看源代码:
1700443540
1700443541
public static<T>List<T>asList(T……a){
1700443542
1700443543
return new ArrayList<T>(a);
1700443544
1700443545
}
1700443546
1700443547
asList方法输入的是一个泛型变长参数,我们知道基本类型是不能泛型化的,也就是说8个基本类型不能作为泛型参数,要想作为泛型参数就必须使用其所对应的包装类型。那前面的例子传递了一个int类型的数组,为什么程序没有报编译错呢?
1700443548
1700443549
在Java中,数组是一个对象,它是可以泛型化的,也就是说我们的例子是把一个int类型的数组作为了T的类型,所以转换后在List中就只有一个类型为int数组的元素了,我们打印出来看看,代码如下:
1700443550
1700443551
public static void main(String[]args){
1700443552
1700443553
int[]data={1,2,3,4,5};
1700443554
1700443555
List list=Arrays.asList(data);
1700443556
1700443557
System.out.println(“元素类型:”+list.get(0).getClass());
1700443558
1700443559
System.out.println(“前后是否相等:”+data.equals(list.get(0)));
1700443560
1700443561
}
1700443562
1700443563
输出的结果是:
1700443564
1700443565
元素类型:class[I
1700443566
1700443567
前后是否相等:true
1700443568
1700443569
很明显,放在列表中的元素是一个int数组,可能有读者要问了,为什么“元素类型:”后的class是“[I”?我们并没有指明是数组(Array)类型呀!这是因为JVM不可能输出Array类型,因为Array是属于java.lang.reflect包的,它是通过反射访问数组元素的工具类。在Java中任何一个数组的类都是“[I”,究其原因就是Java并没有定义数组这一个类,它是在编译器编译的时候生成的,是一个特殊的类,在JDK的帮助中也没有任何数组类的信息。
[
上一页 ]
[ :1.70044352e+09 ]
[
下一页 ]