【Java核心知识】泛型和类型擦除
文章目录
- 泛型
- 什么是泛型
- 类型限定
- 类型擦除
- 如何在运行时判断泛型具体类型
- 参考链接
泛型
什么是泛型
Java中的泛型
是通过定义模板参数来处理一类操作,这类操作并不关心具体传入的参数类型
。比如对于add()
方法来说,我们可以使两个int
相加,也可以使两个long
相加,如果不使用泛型,因为函数签名是通过参数类型区分的,那么就会造成需要写两个方法的问题.
但这两个方法内部的逻辑是一致的,只是传入参数不相同,况且方法内部也并不关心你传入的参数类型。
针对这种情况,我们可以使用泛型
解决,泛型并不会限定你传入的参数类型,它只关心方法内部的逻辑操作
。
常见的集合类
都使用了泛型
。
类型限定
既然泛型类可以传入各种各样的类型,那么使用上会不会出问题呢?比如一个集合既放入了Integer
,又放入了String
。
答案是不会的
,我们在创建泛型类时,一般都会提前传入泛型的类型进行类型限定,比如 Array<Integer> list = new ArrayList<>();
,表明list只能存放Integer
类型的对象,而不能存放String
类型的变量,这是由于编译器在编译代码时通过类型限定对代码进行了检查,从而避免传入不一致的类型;
类型擦除
上面我们讲到是在编译时进行了类型检查,其实Java的泛型并不是真正的泛型,编译完成后会进行类型擦除,也就是说ArrayList
这个类并不知道自己存放的类型是什么,内部统一用基类Object
表示,这就是类型擦除
。
如果正常写代码,类型擦除对我们并没有什么影响,可如果想要通过反射获取泛型属性的具体类型是做不到的。因为反射时我们一般先拿到对应类的Class对象
,然后利用反射获取Field
,但由于类型擦除的存在,泛型属性在Class
对象内部都是基类Object
,并不是具体的类型。
下面是一个示例,count
是一个泛型属性,如果想要通过反射获取count
运行时真正的类型,是做不到的,type
永远是一个java.lang.Object
。
public class ValJson {
static class Count<T> {
public T count;
public void setCount(T count) {
this.count = count;
}
}
public static void main(String[] args) throws NoSuchFieldException {
Count<Integer> count = new Count<>();
count.setCount(2);
// 由于类型擦除,type 的类型为 java.lang.Object
Type type = count.getClass().getField("count").getType();
System.out.println(count);
}
}
如何在运行时判断泛型具体类型
只能在方法内部通过isinstance(a, String.class)
逐一判断类型实现,这样会有很多种if语句
。
参考链接
Java泛型类型擦除以及类型擦除带来的问题
还没有评论,来说两句吧...