public Son extends Father<User>{
.....
}
该泛型类Father<T>的代码如下:
public Father(){
Class c = this.getClass();
Type t = c.getGenericSuperclass();
if(t instance of ParameterizedType)
{
// System.out.println("in if");
Type[] p =(ParameterizedType) t.getActualTypeArguments();
// System.out.println(Arrays.toString(p));
this.class =(Class<T>) p[0];
}
}
Father<T>的构造函数中的 this 指的是子类的实例。
this.getClass()获取子类的真实类型;
c.getGenericSuperclass(); 获取泛型父类 ;
t.getActualTypeArguments();
通过该泛型父类来获取真实的类型参数的数组;也就是获取泛型父类的尖括号里面的参数<T>, 因为泛型的尖括号是可以有多个参数的,所以该方法返回的是一个数组。
p[0](既:t.getActualTypeArguments()[0]); 获取该数组的第一个值。因为我们知道在该例子中Father<T>只有一个参数T。所以我们只需要获取第一个值就可以了。
在这里需要注意的是:范式应该在编译的时候就指定,而不是运行时。
如果我把上面Father(){ }改成如下这样子:
[错误示范]
public Father(){
ParameterizedType t =(ParameterizedType)this.getClass().getGenericSuperclass();
this.class =(Class<T>) t.getActualTypeArguments()[0]; //输出结果,验证类型名称是否正确
//System.out.println("class --> " + class.getSimpleName()); //getSimpleName()是返回不包含包名的类名称
}
上面这段代码,因为在编译器进行编译的时候,还不能确定泛型的具体类型,所以会报如下的 误:java.lang.ClassCastException: java.lang.Class cannot be cast to java.lang. reflect. ParameterizedType。