Java反射中的泛型类型推断

不少框架或者项目的工具类中,有通过反射获取泛型的类型,确实是很常用的代码,但有容易忽略的问题,我在这里总结。

public class T1<T> {
    private Class classt;
    public T1() {
        ParameterizedType type = (ParameterizedType) this.getClass().getGenericSuperclass();
        System.out.println("泛型<T>是否为Class:" + (type.getActualTypeArguments()[0] instanceof Class));
        this.classt = (Class) type.getActualTypeArguments()[0];
        System.out.println(this.classt);
    }
}

1. 父子类的泛型类型都不确定

public class T2<T> extends T1<T> {
    public static void main(String[] args) {
        T2 t = new T2();
    }
}

结果:
Exception in thread “main” java.lang.ClassCastException: sun.reflect.generics.reflectiveObjects.TypeVariableImpl cannot be cast to java.lang.Class
泛型是否为Class:false
2. 父类、子类的泛型类型都确定,且相同

public class T3<Integer> extends T1<Integer> {
    public static void main(String[] args) {
        T3 t = new T3();
    }
}

结果:
Exception in thread “main” java.lang.ClassCastException: sun.reflect.generics.reflectiveObjects.TypeVariableImpl cannot be cast to java.lang.Class
泛型是否为Class:false
3. 父类的泛型类型确定,子类的泛型类型省略

public class T4 extends T1<Integer> {
    public static void main(String[] args) {
        T4 t = new T4();
    }
}

结果:
泛型是否为Class:true
class java.lang.Integer
4. 父类、子类的泛型类型都确定,且相同,父类指定全路径,子类简写

public class T5<Integer> extends T1<java.lang.Integer> {
    public static void main(String[] args) {
        T5 t = new T5();
    }
}

结果:
泛型是否为Class:true
class java.lang.Integer


总结:
第一种情况,因为无法推断具体类型,所以只能是TypeVariable。
第二种情况,因为父子类指定类型相同,导致Integer当做是泛型变量的变量名处理了,而不是java.lang.Integer类,等同于第一种情况。
第三种情况,父类指定了泛型类型,子类泛型类型必须和父类一致,故而省略,最为正确的写法。
第四种情况,避免了第二种相同的名称,运行结果正确,但写法不建议。

参考文章

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值