Java如何获得泛型类中的泛型类型T.class

 更新时间:2024年12月14日 15:37:05   作者:nvd11  
获取泛型符号E的Class对象的方法,1.反射方案利用匿名子类实例化,但该方法有特殊限制,2.反射方案2直接指定泛型类型,但不灵活,3.构造方法方案在初始化时传入泛型类,但有写错风险

Java获得泛型类的泛型类型T.class

假如我们写了1个泛型类MyBase

public class MyBase<E> {
    public MyBase(){
        
    }

    public Class<?> getEClass() {
		?
    }   
}

我想写1个getEClass 的方法, 获取泛型符号E的Class对象。

直接用E.class E.getClass()都是不合法的, 因为E只是个符号, 没有被实例化。

1. 反射方案

各种google 各种Stackoverflow 后, 找到利用反射的1个方法。

public class MyBase<E> {
    public MyBase(){
        
    }

    public Class<?> getEClass() {
        
        //get the Class object of this own class 
        Class<? extends MyBase> thisClass = this.getClass();

        //get the Type Object of supper class
        Type superClassType = thisClass.getGenericSuperclass();
        ParameterizedType pt = (ParameterizedType)superClassType;

        //get the Generic Type array
        Type[] genTypeArr = pt.getActualTypeArguments();
        Type genType = genTypeArr[0];
        if (false == genType instanceof Class){
            return Object.class;
        }

        return (Class<Object>) genType;
    }
}

这个方法被测试过, 但是有个很特别的限制,

一定要实例化匿名子类才有效果。

例如:

   @Test
    void testMyBase() {
        MyBase<String> mybase = new MyBase<String>(){};
        log.info("E class is {}",mybase.getEClass());
    }

注意new MyBase(){} 后面的大括号, 意思不是实例化MyBase对象, 而是实例化MyBase的匿名子类。

如果我们写多1个具体的子类MyList

public class MyList<E> extends MyBase<E> {
    public MyList(){

	}
}

测试方法:

   @Test
    void testGetEClass() {
        MyList<String> mylist = new MyList<String>();
        log.info("E class is {}",mylist.getEClass());
    }

同样不行, 原因是

		//get the Type Object of supper class
        Type superClassType = thisClass.getGenericSuperclass();
        ParameterizedType pt = (ParameterizedType)superClassType;

这里 我们只能通过本类的Class对象去获得1个父类 ParameterizedType (包括泛型参数列表的Type)

而JDK本身只提供了getGenericSuperclass()方法去获得父类的Type 对象

但是没有提供例如getGenericClass() 去获取本类的Type对象…

所以一旦我们不实用匿名子类

当下面代码被执行时

MyList<String> mylist = new MyList<String>();
log.info("E class is {}",mylist.getEClass());

thisClass.getGenericSuperclass() 会获得MyList的父类MyBase的Type, 问题是MyBase没有被实例化, 所以pt.getActualTypeArguments()[0] 只会返回E

本身 就return null了

但是如果我们利用匿名子类的化

当下面代码被执行时

MyList<String> mylist = new MyList<String>(){};
log.info("E class is {}",mylist.getEClass());

mylist 的类实际上是1个匿名子类MyList$1, 它的父类就是MyList, 而匿名子类的父类在JVM实际上是被实例化的

所以pt.getActualTypeArguments()[0] 会返回泛型具体类String 的Class。

2. 反射方案2

匿名子类毕竟有点奇怪。

如果避免使用你匿名子类, 我们还有1个办法

public class MySet extends MyBase<String> {
    public MySet(){
  
    }
}

如上面这个MySet类, 它直接继承MyBase时指定了具体泛型, 令到自己本身不是1个泛型类。

这样的话, 不用匿名子类也一样可以获得泛型具体类型。

因为这种情况下, 一旦实现MySet, 因为MySet指定了继承泛型的具体类型, 在JVM 中MyBase也会被实例化。

但是这个方案一样不好, 不灵活是其次, 我都会指定了具体泛型类型了, 何必用再用getEClass()去获得泛型具体类型? 多次一举

  @Test
    void testGetEClass2() {
        MySet myset = new MySet();
        log.info("Set E class is {}",myset.getEClass());
    }

3. 构造方法方案

这也是外网强烈推荐的方案

public class MyMap<E> {

    private Class<E> eClass;

    public MyMap(Class<E> eClass){
        this.eClass = eClass;
    }

    public Class<E> getEClass(){
        return this.eClass;
    }

}

让开发人员在初始化时,顺便外部传入泛型的类, 不过手抖写错的风险…

测试方法:

  @Test
    void testGetEClass3() {
        MyMap<String> myMap= new MyMap<>(String.class);
        log.info("Set E class is {}",myMap.getEClass());
    }
}

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • Java 按照字节来截取字符串的代码(不会出现半个汉字)

    Java 按照字节来截取字符串的代码(不会出现半个汉字)

    Java 按照字节来截取字符串的工具,不会出现半个汉字。一个中文两个字节,一个英文字符只占 1 个字节** 1. 通常我们用于前端显示的时候,防止标题过长
    2014-01-01
  • SpringCloud将Nacos作为配置中心实现流程详解

    SpringCloud将Nacos作为配置中心实现流程详解

    这篇文章主要介绍了Springcloud中的Nacos Config服务配置,本文以用户微服务为例,进行统一的配置,结合实例代码给大家介绍的非常详细,需要的朋友可以参考下
    2022-10-10
  • Spring context:component-scan的使用及说明

    Spring context:component-scan的使用及说明

    这篇文章主要介绍了Spring context:component-scan的使用及说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-09-09
  • Java读取传输FTP文件实现示例

    Java读取传输FTP文件实现示例

    本文主要介绍了Java读取传输FTP文件方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-04-04
  • java Socket UDP实例详解

    java Socket UDP实例详解

    这篇文章主要介绍了java Socket UDP实例详解的相关资料,需要的朋友可以参考下
    2017-02-02
  • Java弹簧布局管理器使用方法详解

    Java弹簧布局管理器使用方法详解

    这篇文章主要介绍了Java弹簧布局管理器使用方法详解,需要的朋友可以参考下
    2017-09-09
  • SpringBoot自定义Banner使用详解

    SpringBoot自定义Banner使用详解

    这篇文章主要介绍了SpringBoot自定义Banner使用详解,启动 Spring Boot 时,几乎总是能在控制台上方看到如下横幅,这个也叫字符画、英文ASCII艺术字,这就是banner,我们来看一下如何使用,需要的朋友可以参考下
    2024-01-01
  • java发送短信系列之同步、异步发送短信

    java发送短信系列之同步、异步发送短信

    这篇文章主要介绍了java发送短信系列之同步、异步发送短信的相关资料,感兴趣的小伙伴们可以参考一下
    2016-02-02
  • Springboot项目对数据库用户名密码实现加密过程解析

    Springboot项目对数据库用户名密码实现加密过程解析

    这篇文章主要介绍了Springboot项目对数据库用户名密码实现加密过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-06-06
  • 分析HashMap 的 JDK 源码

    分析HashMap 的 JDK 源码

    这篇文章主要分析了HashMap 的 JDK 源码,帮助大家更好的理解和学习Java,感兴趣的朋友可以了解下
    2020-10-10

最新评论