文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Java中如何获取泛型类型信息

2023-07-05 09:56

关注

这篇文章主要讲解了“Java中如何获取泛型类型信息”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Java中如何获取泛型类型信息”吧!

根据使用泛型位置的不同可以分为:声明侧泛型、使用侧泛型。

声明侧的泛型信息被记录在Class文件的Constant pool中以Signature的形式保存。而使用侧的泛型信息并没有保存。

声明侧泛型

声明侧泛型包括:

使用侧泛型

使用侧泛型包括:

获取泛型类型相关方法

上文有提到,声明侧的泛型被记录在Class文件的Constant pool中以Signature的形式保存。

JDK的Class、Field、Method类提供了一系列的获取泛型类型的相关方法。

1. Class类的泛型方法

Type getGenericSuperclass():获取父类的Type

Type[] getGenericInterfaces():获取父接口的Type集合

  1. 若父类有泛型,返回的实际Type是ParameterizedType接口的实现类ParameterizedTypeImpl类

  2. 若父类无泛型,返回的实际Type是Class类

2. Field类的泛型方法

Type getGenericType():获取字段的Type

3. Method类的泛型方法

Type getGenericReturnType():获取方法返回值的Type

Type[] getGenericParameterTypes():获取方法参数的Type集合

Type[] getGenericExceptionTypes():获取方法声明的异常的Type集合

4. ParameterizedType类

ParameterizedType是Type的子接口,表示参数化类型,用于获取泛型的参数类型。

ParameterizedType的主要方法:

获取声明侧的泛型类型信息

示例:

public class MyTest extends TestClass<String> implements TestInterface1<Integer>,TestInterface2<Long> {    private List<Integer> list;    private Map<Integer, String> map;    public List<String> aa() {        return null;    }    public void bb(List<Long> list) {    }    public static void main(String[] args) throws Exception {        System.out.println("======================================= 泛型类声明的泛型类型 =======================================");        ParameterizedType parameterizedType = (ParameterizedType)MyTest.class.getGenericSuperclass();        System.out.println(parameterizedType.getTypeName() + "--------->" + parameterizedType.getActualTypeArguments()[0].getTypeName());        Type[] types = MyTest.class.getGenericInterfaces();        for (Type type : types) {            ParameterizedType typ = (ParameterizedType)type;            System.out.println(typ.getTypeName() + "--------->" + typ.getActualTypeArguments()[0].getTypeName());        }        System.out.println("======================================= 成员变量中的泛型类型 =======================================");        ParameterizedType parameterizedType1 = (ParameterizedType)MyTest.class.getDeclaredField("list").getGenericType();        System.out.println(parameterizedType1.getTypeName() + "--------->" + parameterizedType1.getActualTypeArguments()[0].getTypeName());        ParameterizedType parameterizedType2 = (ParameterizedType)MyTest.class.getDeclaredField("map").getGenericType();        System.out.println(parameterizedType2.getTypeName() + "--------->" + parameterizedType2.getActualTypeArguments()[0].getTypeName()+","+parameterizedType2.getActualTypeArguments()[1].getTypeName());        System.out.println("======================================= 方法参数中的泛型类型 =======================================");        ParameterizedType parameterizedType3 = (ParameterizedType)MyTest.class.getMethod("aa").getGenericReturnType();        System.out.println(parameterizedType3.getTypeName() + "--------->" + parameterizedType3.getActualTypeArguments()[0].getTypeName());        System.out.println("======================================= 方法返回值中的泛型类型 =======================================");        Type[] types1 = MyTest.class.getMethod("bb", List.class).getGenericParameterTypes();        for (Type type : types1) {            ParameterizedType typ = (ParameterizedType)type;            System.out.println(typ.getTypeName() + "--------->" + typ.getActualTypeArguments()[0].getTypeName());        }    }}class TestClass<T> {}interface TestInterface1<T> {}interface TestInterface2<T> {}

输出

======================================= 泛型类声明的泛型类型 =======================================
com.joker.test.generic.TestClass<java.lang.String>--------->java.lang.String
com.joker.test.generic.TestInterface1<java.lang.Integer>--------->java.lang.Integer
com.joker.test.generic.TestInterface2<java.lang.Long>--------->java.lang.Long
======================================= 成员变量中的泛型类型 =======================================
java.util.List<java.lang.Integer>--------->java.lang.Integer
java.util.Map<java.lang.Integer, java.lang.String>--------->java.lang.Integer,java.lang.String
======================================= 方法参数中的泛型类型 =======================================
java.util.List<java.lang.String>--------->java.lang.String
======================================= 方法返回值中的泛型类型 =======================================
java.util.List<java.lang.Long>--------->java.lang.Long

获取使用侧的泛型类型信息

上面讲的相关类的获取泛型类型相关方法都只是针对声明侧的泛型。因为声明侧的泛型被记录在Class文件的Constant pool中以Signature的形式保存。所以Java提供了相关方法能获取到这些信息。

那使用侧的泛型信息怎么获取呢?由于使用侧的泛型信息在编译期的时候就被类型擦除了,所以运行时是没办法获取到这些泛型信息的。

难道就真的没办法了吗,其实还是有的。使用侧需要获取泛型信息的地方主要是:方法调用时传入的泛型变量,通常需要在方法中获取变量的泛型类型。比如在JSON解析(反序列化)的场景,他们是怎么实现的了。

针对获取使用侧的泛型类型信息,主要实现方案是通过匿名内部类。

Gson中的泛型抽象类TypeToken<T>,FastJson中的泛型类TypeReference<T>等就是用的该方案。

匿名内部类实现获取使用侧的泛型类型

上文有讲到,在声明侧的泛型中,针对泛型类或泛型接口的声明的泛型,Class类提供了getGenericSuperclass()、getGenericInterfaces()来获取其子类(实现类)上声明的具体泛型类型信息。

而匿名内部类是什么?其本质就是一个继承/实现了某个类(接口,普通类,抽象类)的子类匿名对象。

匿名内部类实现获取使用侧的泛型类型的原理:

简单示例:

定义泛型类TestClass2<T>,类中包含字段Type

public abstract class TestClass2<T> {    private final Type type;    public TestClass2() {        Type superClass = getClass().getGenericSuperclass();        if (!(superClass instanceof ParameterizedType)) {            throw new IllegalArgumentException("无泛型类型信息");        }        type = ((ParameterizedType) superClass).getActualTypeArguments()[0];    }    public Type getType() {        return type;    }}

测试获取泛型类型

public class Test {    public static  <T> T get(TestClass2<T> tTestClass2) throws IllegalAccessException, InstantiationException {        Type type = tTestClass2.getType();        Class clazz = (Class) type;        return (T)clazz.newInstance();    }    public static void main(String[] args) throws InstantiationException, IllegalAccessException {        String str = get(new TestClass2<String>() {});        Date date = get(new TestClass2<Date>() {});    }}

感谢各位的阅读,以上就是“Java中如何获取泛型类型信息”的内容了,经过本文的学习后,相信大家对Java中如何获取泛型类型信息这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是编程网,小编将为大家推送更多相关知识点的文章,欢迎关注!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯