您的位置:首页 > 编程语言 > Java开发

Java_集合操作_避开基本类型数组转换列表陷阱

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

package deep;

import java.util.Arrays;
import java.util.List;

public class Client {

public static void main(String[] args) {
int[] data = { 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<>(a);
}


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

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

package deep;

import java.util.Arrays;
import java.util.List;

public class Client {

public static void main(String[] args) {
int[] data = { 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) {
Integer[] data = { 1, 2, 3, 4, 5 };
List list = Arrays.asList(data);
System.out.println("列表中的元素数量是:" + list.size());
}


把int替换为Integer即可让输出元素数量为5。需要说明的是,不仅仅是int类型的数组有这个问题,其他7个基本类型的数组也存在相似的问题,这就需要读者注意了,在把基本类型数组转换为列表时,要特别小心asList方法的陷阱,避免出现程序逻辑混乱的情况。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  基本类型 java