Java_集合操作_避开基本类型数组转换列表陷阱
2015-04-11 22:04
309 查看
我们在开发过程中经常会使用Arrays和Collections这两个工具类在数组和列表之间转换,非常方便,但也有时候会出现一些奇怪的问题,来看如下代码:
也许你会说,这很简单,list变量的元素当然是5了。但是运行后打印出来的列表数量却是1。
事实上data确实是一个有5个元素的int类型数组,只是通过asList转换成列表后就只有1个元素了,这是为什么呢?其他4个元素到什么地方去了呢?
我们仔细看一下Arrays.asList的方法说明:输入一个变长参数,返回一个固定长度的列表。注意这里是一个变长参数,看源代码:
asList方法输入的是一个泛型变长参数,我们知道基本类型是不能泛型化的,也就是说8个基本类型不能作为泛型参数,要想作为泛型参数就必须使用其所对应的包装类型。那前面的例子传递了一个int类型的数组,为什么程序没有报编译错误呢?
在Java中,数组是一个对象,它是可以泛型化的,也就是说我们的例子是把一个int类型的数组作为了T的类型,所以转换后在List中就只有一个类型为int类型的元素了,我们打印出来看看,代码如下:
运行结果:
元素类型:class [I
前后是否相等:true
很明显,放在列表中的元素是一个int数组,可能有读者要问了,为什么“元素类型:”后的class是“[I”?我们并没有指明是数组(Array)类型呀!这是因为JVM不可能输出Array类型,因为Array是属于java.lang.reflect包的,它是通过反射访问数组元素的工具类。在Java中任何一个数组的类都是“[I”,究其原因就是Java并没有定义数组这一个类,它是在编译器编译的时候生成的,是一个特殊的类,在JDK的帮助中也没有任何数组类的信息。
弄清楚了问题,修改方案也就诞生了,直接使用包装类即可,代码如下:
把int替换为Integer即可让输出元素数量为5。需要说明的是,不仅仅是int类型的数组有这个问题,其他7个基本类型的数组也存在相似的问题,这就需要读者注意了,在把基本类型数组转换为列表时,要特别小心asList方法的陷阱,避免出现程序逻辑混乱的情况。
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程序的151个建议]建议65 避开基本类型数组转换列表陷阱
- [改善Java代码]避开基本类型数组转换列表陷阱
- 避开基本类型数组转换列表陷阱
- 提高你的Java代码质量吧:避免基本类型数组转换列表陷阱(Arrays.asList())
- 避开基本类型数组转换列表陷阱
- java避开基本数据类型转换列表陷阱
- Java中的移位操作以及基本数据类型转换成字节数组【收集】
- Java中的移位操作以及基本数据类型转换成字节数组【收集】
- Java基础小知识——基本数据类型和字符串以及字节数组的转换
- Java基础小知识——基本数据类型和字符串以及字节数组的转换
- Java基本类型与byte数组之间相互转换
- Java基本类型与byte数组之间相互转换
- java基本数据类型转换成byte[]数组
- java基本数据类型转换成byte[]数组
- Java基本类型与byte数组之间相互转换
- Java -- 利用反射 操作任意数组,包括对象数组 和 基本数据类型的数组
- Java -- 利用反射 操作任意数组,包括对象数组 和 基本数据类型的数组
- Java中字节数组与基本数据类型的转换
- JAVA基础学习之String、StringBuffer、StringBuilder、基本数据类型的使用、整形进制转换、集合Collection、Vector、ArrayList、LinkedList、HashSet、TreeSet等(3)