概述
泛型在java中有很重要的地位,在面向对象编程及各种设计模式中有非常广泛的应用。
什么是泛型?为什么要使用泛型?
泛型,即“参数化类型”。一提到参数,最熟悉的就是定义方法时有形参,然后调用此方法时传递实参。那么参数化类型怎么理解呢?顾名思义,就是将类型由原来的具体的类型参数化,类似于方法中的变量参数,此时类型也定义成参数形式(可以称之为类型形参),然后在使用/调用时传入具体的类型(类型实参)。
泛型的本质是为了参数化类型(在不创建新的类型的情况下,通过泛型指定的不同类型来控制形参具体限制的类型)。也就是说在泛型使用过程中,操作的数据类型被指定为一个参数,这种参数类型可以用在类、接口和方法中,分别被称为泛型类、泛型接口、泛型方法。
List arrayList = new ArrayList();
arrayList.add("aaaa"); // 不会报错
arrayList.add(100); //不会报错
for(int i = 0; i< arrayList.size();i++){
String item = (String)arrayList.get(i); // 会报错,编译错误
Log.d("泛型测试","item = " + item);
}
毫无疑问。这个程序就肯定会报错,
java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.String
因为 arrayList 没有定义具体类型,而存储了一个int类型的,一个string类型的,循环输出用string接收,get(0)="aaaa",而get(1)=100;get(1)是int类型,所以会在程序运行时报错
那么我们将第一行声明初始化list的代码更改一下,编译器会在编译阶段就能够帮我们发现类似这样的问题。
List<String> arrayList = new ArrayList<String>();
arrayList.add("aaaa"); //不会报错
arrayList.add(100); //直接提示编译错误
for(int i = 0; i< arrayList.size();i++){
String item = (String)arrayList.get(i);
Log.d("泛型测试","item = " + item);
}
通过上面的例子可以证明,在编译之后程序会采取去泛型化的措施。也就是说Java中的泛型,只在编译阶段有效。在编译过程中,正确检验泛型结果后,会将泛型的相关信息擦出,并且在对象进入和离开方法的边界处添加类型检查和类型转换的方法。也就是说,泛型信息不会进入到运行时阶段。
对此总结成一句话:泛型类型在逻辑上可以看成是多个不同的类型,实际上都是相同的基本类型。