java怎样动态获取泛型参数的类型

 更新时间:2024年09月19日 10:30:55   作者:Android海纳百川  
在Java中,泛型信息在编译时会被擦除,但可以通过特定API获取运行时的泛型参数类型,主要API包括Class的getGenericSuperclass()和getGenericInterfaces()方法,以及ParameterizedType的getActualTypeArguments()方法

Java如何动态获取泛型参数的类型

我们都知道java中的泛型其实是伪泛型,java在编译阶段会对变异类型进行擦除,擦出到泛型类的最小上限,编译后得到的class文件里面是没有任何泛型信息的,泛型的控制其实就是java编译器进行的控制,编译阶段进行泛型检查。

那如果我们想在运行时知道泛型类的类型,如何做到呢?

主要用到下面几个api:

1、public Type getGenericSuperclass()---Class类的方法

2、Type[] getActualTypeArguments()---ParameterizedType方法

3、public Type[] getGenericInterfaces()---Class类的方法

  • 我们先定义一个泛型类:
public class Father<T> {
}
  • 再定义一个子类:
public class Son extends Father<String>{
}
  • 先来看看下面代码:
System.out.println(Son.class.getGenericSuperclass());
System.out.println(Son.class.getSuperclass());
System.out.println(Father.class.getGenericSuperclass());
System.out.println(Father.class.getSuperclass());

运行结果如下:

从上面运行结果可以看出两点:

(1)通过Class类的getGenericSuperclass()是可以获取到泛型信息的;

(2)泛型类本身是无法获取到泛型信息的,只能通过泛型类的子类来获取泛型信息,这也很好解释,因为泛型类比如Father类自身,其泛型参数并未确定,自然无法获取泛型信息,而子类的泛型参数类型已经确定,父类泛型信息已经确定,所以可以查询。

具体的查询泛型参数类型的代码如下:

ParameterizedType parameterizedType = (ParameterizedType) Son.class.getGenericSuperclass();
System.out.println(parameterizedType.getClass().getName());
System.out.println(parameterizedType);
Type[] actualTypeArguments = parameterizedType.getActualTypeArguments();
for(Type actualTypeArgument: actualTypeArguments) {
    System.out.println(actualTypeArgument.getClass().getName());  
	System.out.println(actualTypeArgument); 
} 

运行结果如下:

由上图运行结果可知:

(1)getGenericSuperclass()返回的实际类型为ParameterizedTypeImpl,这里面带有父类的泛型类型信息;

(2)获取泛型参数信息是通过ParameterizedType的getActualTypeArguments方法,该方法返回的是一个Type类型的数据,该数据存放的数据的实际类型为Class,也就是我们得到了泛型参数的类信息。

上面讲述的是获取父类的泛型类型,那接口的泛型类型如何获取呢?

  • 先贴一个简单的接口:
public interface ITest<T> {
}
  • 我们让Son类实现该方法:
public class Son extends Father<String> implements ITest<Integer>{
}
  • 获取ITest的泛型参数类型的方法如下:
ParameterizedType parameterizedType = (ParameterizedType) Son.class.getGenericInterfaces()[0];
System.out.println(parameterizedType.getClass().getName());
System.out.println(parameterizedType);
Type[] actualTypeArguments = parameterizedType.getActualTypeArguments();
for(Type actualTypeArgument: actualTypeArguments) { 
	System.out.println(actualTypeArgument.getClass().getName());
	System.out.println(actualTypeArgument); 
} 

运行结果如下:

和泛型类区别就是getGenericSuperclass()方法换成了getGenericInterfaces(),getGenericInterfaces()方法返回的是Type[],因为一个类可以实现多个接口,所以想要获取哪个接口的泛型信息,需要指定数据下标,这里Son类就实现了一个接口,所以直接Son.class.getGenericInterfaces()[0]就可以了。

上面的代码有一个共同点

就是泛型类或者泛型接口已经有一个子类了,通过子类的Class信息可以获取到泛型父类或者泛型接口的泛型类型信息,那么如果没有子类,怎么直接获取泛型类的泛型类型信息呢?

很简单,通过匿名内部类(其实还是通过子类).

new Father<Integer>() {}.getClass().getGenericSuperclass();
new ITest<Integer>() {}.getClass().getGenericInterfaces())[0];

总结

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

相关文章

  • Java中的可重入锁ReentrantLock简析

    Java中的可重入锁ReentrantLock简析

    这篇文章主要介绍了Java中的可重入锁ReentrantLock简析,可重入是指同一个线程如果首次获得了这把锁,那么因为它是这把锁的拥有者,因此有权利再次获取这把锁如果是不可重入锁,那么第二次获得锁时,自己也会被锁挡住,需要的朋友可以参考下
    2023-12-12
  • springboot与vue详解实现短信发送流程

    springboot与vue详解实现短信发送流程

    随着人工智能的不断发展,机器学习这门技术也越来越重要,很多人都开启了学习机器学习,本文就介绍了机器学习的基础内容
    2022-06-06
  • Spring Boot中操作使用Redis实现详解

    Spring Boot中操作使用Redis实现详解

    Spring Boot与Redis结合使用,通过使用Spring Data Redis来实现对Redis的操作,实现数据缓存和高效存储,提高应用程序的性能和响应速度。可以利用Spring Boot自带的Redis Starter方便地集成和配置Redis
    2023-04-04
  • 一篇文章带你入门java运算符

    一篇文章带你入门java运算符

    这篇文章主要介绍了Java基本数据类型和运算符,结合实例形式详细分析了java基本数据类型、数据类型转换、算术运算符、逻辑运算符等相关原理与操作技巧,需要的朋友可以参考下
    2021-08-08
  • Java 开发的几个注意点总结

    Java 开发的几个注意点总结

    这篇文章主要介绍了Java开发的几个注意点的相关资料,需要的朋友可以参考下
    2016-09-09
  • SpringBoot Admin使用及心跳检测原理分析

    SpringBoot Admin使用及心跳检测原理分析

    这篇文章主要介绍了SpringBoot Admin使用及心跳检测原理分析,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-11-11
  • SpringBoot Session共享实现图解

    SpringBoot Session共享实现图解

    这篇文章主要介绍了SpringBoot Session共享实现图解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-01-01
  • 解决java连接zookeeper很慢的问题

    解决java连接zookeeper很慢的问题

    这篇文章主要介绍了解决java连接zookeeper很慢的问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-11-11
  • 使用IntelliJ IDEA 2017.2.5 x64中的Spring Initializr插件快速创建Spring Boot/Cloud工程(图解)

    使用IntelliJ IDEA 2017.2.5 x64中的Spring Initializr插件快速创建Spring

    这篇文章主要介绍了使用IntelliJ IDEA 2017.2.5 x64中的Spring Initializr插件快速创建Spring Boot/Cloud工程(图解),需要的朋友可以参考下
    2018-01-01
  • Java实现前缀树详解

    Java实现前缀树详解

    Java实现前缀树(Trie树)是一种树形数据结构,用于字符串的存储和查找,适用于大量字符串的快速匹配。通过将字符串拆分为字符序列,依次构建树形结构,将每个字符串的字符依次存储在树的节点上,实现高效的字符串匹配
    2023-04-04

最新评论