本文链接:https://blog.csdn.net/hongxingxiaonan/article/details/49202613

在Java反射技术的应用中,取得一个类的成员、方法和构造函数相对比较容易,但是要想取得其中包含的泛型类型则相对困难一些。先看一个简单的例子,然后详细说明每个步骤的作用。
public class Demo {
private List<Integer> list1;
private List<Set<Integer>> listSet;
public Set<String> fun1(Map<Integer, String> map){return null;}

public static void getFieldGenericType() {
try {
Class clazz = Demo.class;
Field field = clazz.getDeclaredField("list1");
Type type = field.getGenericType(); //取得field的type
ParameterizedTypeImpl parameterizedType = (ParameterizedTypeImpl) type; //强转成具体的实现类
Type[] genericTypes = parameterizedType.getActualTypeArguments(); //取得包含的泛型类型
System.out.println(genericTypes[0]);
} catch (NoSuchFieldException e) {
e.printStackTrace();
}
}

public static void main(String[] args) {

getFieldGenericType();
}
}
        上面的代码可以正确的打印出Demo类的成员list1的泛型类型Integer。代码中的第一步通过getGenericType()方法获得一个Type类型。这个Type其实就代表了List<Interger>,Type的实现类可以是Class,也可以是ParameterizedTypeImpl。如果field的类型是泛型则通过getGenericType()取到的就是ParameterizedTypeImpl,如果是普通的类则取到的就是Class。我们在编译器取到的Class是没有泛型信息的,通过field的getType方法取到的信息只能是List,而不是List<Interger>,所以我们才用上面的方法获取泛型的类型;第二步将Type强转成了ParameterizedTypeImpl类型,这里偷了个懒,因为知道真实的类型所以没有判断type的具体实现类,开发时千万不能这样写;第四步从ParameterizedTypeImpl中取得到了所有的泛型类型。ParameterizedTypeImpl也就是参数化的type,我们要获得的泛型类型相当于type的参数,有几个泛型类型,则getActualTypeArguments()会返回几个Type参数。
        刚才说到Class是Type的子类,而通过getActualTypeArguments()获取到的又是一个Type数组,好像陷入到了无限的循环当中。没错,在真实的代码中也确实存在着泛型类型的嵌套使用,例如上面代码中的第二个域listSet。有了刚才的基础,获取多少层的嵌套泛型也可以解析出来。解析listSet中泛型信息的代码如下:
public static void getFieldGenericType1() {
try {
Class clazz = Demo.class;
Field field = clazz.getDeclaredField("listSet");
Type type = field.getGenericType(); //取得field的type
ParameterizedTypeImpl parameterizedType = (ParameterizedTypeImpl) type; //强转成具体的实现类
Type[] genericTypes = parameterizedType.getActualTypeArguments(); //取得包含的泛型类型

ParameterizedTypeImpl setType = (ParameterizedTypeImpl) genericTypes[0];//再次将代表Set<Integer>的Type强转成ParameterizedTypeImpl
Type[] setTypeArguments = setType.getActualTypeArguments();
System.out.println(setTypeArguments[0]);
} catch (NoSuchFieldException e) {
e.printStackTrace();
}
}
         同样的道理,获取方法中的泛型信息也不在话下
public static void getMethodGenericType() {
try {
Class clazz = Demo.class;
Method method = clazz.getMethod("fun1", Map.class);
Type type = method.getGenericReturnType();
ParameterizedTypeImpl parameterizedType = (ParameterizedTypeImpl) type;
Type[] genericTypes = parameterizedType.getActualTypeArguments();
System.out.println("return generic type " + genericTypes[0]);

Type mapType = method.getGenericParameterTypes()[0];
ParameterizedTypeImpl mapParamType = (ParameterizedTypeImpl) mapType;//再次将代表Set<Integer>的Type强转成ParameterizedTypeImpl
Type[] mapArgs = mapParamType.getActualTypeArguments();
System.out.println("method param first generic type " + mapArgs[0]);
} catch (Exception e) {
e.printStackTrace();
}
}
        上面展示了解析成员和方法中泛型信息的方法,但是对于一个普通的变量可以在运行期解析出他的泛型信息吗?答案是不可以的,因为对于一个普通的变量我们只能拿到的他的Class信息,而刚刚也已经说到Class是没有泛型信息的。如果要获取到泛型的信息,首先要获取到ParameterizedTypeImpl才可以。
————————————————
版权声明:本文为CSDN博主「hongxingxiaonan」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/hongxingxiaonan/article/details/49202613

Java反射获取泛型类型的更多相关文章

  1. Java 反射 -- 获取泛型类型

    先写一个类: public class Demo03 { public void test01(Map<String, User> map, List<User> list) ...

  2. Java反射获取对象成员属性,getFields()与getDeclaredFields()方法的区别

    Java反射获取对象成员属性,getFields()与getDeclaredFields()方法的区别 ​ 在工作中遇到一个问题,就是你需要去判断某个字符串是不是对象的某个成员属性名,然后根据判断结果 ...

  3. Java反射获取class对象的三种方式,反射创建对象的两种方式

    Java反射获取class对象的三种方式,反射创建对象的两种方式 1.获取Class对象 在 Java API 中,提供了获取 Class 类对象的三种方法: 第一种,使用 Class.forName ...

  4. java反射获取注解并拼接sql语句

    先建两个注解 分别为 Table 和 Column package com.hk.test; import java.lang.annotation.ElementType; import java. ...

  5. 关于java反射获取泛型

    public class Test<T> { private final TypeToken<T> typeToken = new TypeToken<T>(get ...

  6. Java 反射获取私有方法

    通常我们创建一个类时,它的私有方法在类外是不可见的,但是可以通过反射机制来获取调用.具体的反射机制的介绍大家自己百度. 所以反射可能会破坏我们的单例模式,当然解决方案也是有的,就是做个标记记录次数,第 ...

  7. java反射获取和设置实体类的属性值 递归所有父类

    最近做一个通用数据操作接口,需要动态获取和设置实体类的属性值,为了通用实体做了多重继承,开始网上找到代码都不支持父类操作,只能自己搞一个工具类了,此工具类可以设置和获取所有父类属性,代码贴下面拿走不谢 ...

  8. java反射获取Object的属性和值

    在看反射顺便做个笔记,目前知道的反射的Object都是要有对象的也就是实体Bean. import java.lang.reflect.Field; import java.util.ArrayLis ...

  9. java 反射获取字段为List类型中的泛型类型

    在java编程时,经常会使用反射去处理对象,有时会需要知道List的泛型类型:示例操作如下 obj = pojo.newInstance(); // 首先得到pojo所定义的字段 Field[] fi ...

随机推荐

  1. hadoop 完全分布式集群搭建

    1.在伪分布式基础上搭建,伪分布式搭建参见VM上Hadoop3.1伪分布式模式搭建 2.虚拟机准备,本次集群采用2.8.3版本与3.X版本差别不大,端口号所有差别 192.168.44.10 vmho ...

  2. xshell5运行hadoop集群

    ---恢复内容开始--- 1.CentOS主机配置 在配置Hadoop过程中,防火墙必须优先关闭SELinux,否则将影响后续Hadoop配置与使用,命令如下: # 查看 “系统防火墙” 状态命令 s ...

  3. Redis数据库详解

    NoSQL 若杀死进程应使用pkill 数据设计模式:分布式.非关系型.不提供ACID 特性:简单数据模型.源数据和应用数据分离.弱一致性 优势: 避免不必要的复杂性 高吞吐量, 高 水平扩展能力和低 ...

  4. 【MySQL高可用架构设计】(一)-- mysql复制功能介绍

    一. 介绍 Mysql的复制功能是构建基于SQL数据库的大规模高性能应用的基础,主要用于分担主数据库的读负载,同时也为高可用.灾难恢复.备份等工作提供了更多的选择. 二.为什么要使用mysql复制功能 ...

  5. 笔记5:Django知识一

    Django 1 MVC MVC的核心思想就是解耦. 2 Django介绍 2.1 MVT Django遵循MVC思想,其称为MVT.其中: M: Model模型:和MVC中的M相同和数据库交互 V: ...

  6. django中嵌套的try-except

    # 因为此时为yaml模板,而且只抓取node port,所以这样处理效率快content_dict = parse_yaml(content.replace("{{", &quo ...

  7. echo和printf打印输出

    [root@node2 scprits]# echo Hello World! Hello World! [root@node2 scprits]# echo 'Hello World!' Hello ...

  8. nginx发布静态资源

    nginx发布静态资源 参考 ngx_http_index_module index指令 ngx_http_core_module http指令 location指令 listen指令 root指令 ...

  9. 20191003 「HZOJ NOIP2019 Round #8」20191003模拟

    综述 试题为常州集训2019SCDay2 得分\(100+30(0)+28\) 时之终结 问题描述 HZOJ1310 题解 构造题. 发现部分分有一档是 \(Y\) 是 \(2^x\) ,于是自然想到 ...

  10. windwos文件句柄数限制

    1.修改注册表,位置如下: HKEY_LOCAL_MACHINE/SOFTWARE/Microsoft/Windows NT/CurrentVersion/Windows 2.设置 1.GDIProc ...