返回顶部
首页 > 资讯 > 后端开发 > Python >Java中获取泛型类型信息的方法
  • 321
分享到

Java中获取泛型类型信息的方法

Java获取泛型类型信息Java泛型类型信息 2023-03-08 11:03:23 321人浏览 泡泡鱼

Python 官方文档:入门教程 => 点击学习

摘要

目录声明侧泛型使用侧泛型获取泛型类型相关方法1. Class类的泛型方法2. Field类的泛型方法3. Method类的泛型方法4. ParameterizedType类获取使用侧

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

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

声明侧泛型

声明侧泛型包括:

  • 泛型类,或泛型接口的声明
  • 带有泛型参数的成员变量
  • 带有泛型参数的方法

使用侧泛型

使用侧泛型包括:

  • 方法的局部变量,
  • 方法调用时传入的变量

获取泛型类型相关方法

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

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

1. Class类的泛型方法

Type getGenericSuperclass():获取父类的Type

  • 若父类有泛型,返回的实际Type是ParameterizedType接口的实现类ParameterizedTypeImpl类
  • 若父类无泛型,返回的实际Type是Class类

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

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

2. Field类的泛型方法

Type getGenericType():获取字段的Type

  • 若字段有泛型,返回的实际Type是ParameterizedType接口的实现类ParameterizedTypeImpl类
  • 若字段无泛型,返回的实际Type是Class类

3. Method类的泛型方法

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

  • 若返回值有泛型,返回的实际Type是ParameterizedType接口的实现类ParameterizedTypeImpl类
  • 若返回值无泛型,返回的实际Type是Class类

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

  • 若方法参数有泛型,返回的实际Type是ParameterizedType接口的实现类ParameterizedTypeImpl类
  • 若方法参数无泛型,返回的实际Type是Class类

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

  • 若方法参数有泛型,返回的实际Type是ParameterizedType接口的实现类ParameterizedTypeImpl类
  • 若方法参数无泛型,返回的实际Type是Class类

4. ParameterizedType类

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

ParameterizedType的主要方法:

  • Type[] getActualTypeArguments():获取实际类型参数的Type集合
  • Type getRawType():获取声明此类型的类或接口的Type
  • Type getOwnerType():如果声明此类型的类或接口为内部类,这返回的是该内部类的外部类的Type(也就是该内部类的拥有者)

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

  • 泛型类,或泛型接口的声明
  • 带有泛型参数的成员变量
  • 带有泛型参数的方法

示例:

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()来获取其子类(实现类)上声明的具体泛型类型信息。

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

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

  • 定义泛型类,泛型类中有一个Type类型的字段,用于保存泛型类型的Type
  • 通过匿名内部类的方式创建该泛型类的子类实例(指定了具体的泛型类型)
    在创建子类实例的构造方法中,已经通过子类的Class的getGenericSuperclass()获取到了泛型类型信息并复制给了Type类型的字段中。
  • 随后任何地方,只要得到了该子类实例,就可以通过实例得到泛型类型的Type,这就得到了使用侧的泛型类信息。

简单示例:

定义泛型类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获取泛型类型信息内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: Java中获取泛型类型信息的方法

本文链接: https://lsjlt.com/news/198971.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • Java中获取泛型类型信息的方法
    目录声明侧泛型使用侧泛型获取泛型类型相关方法1. Class类的泛型方法2. Field类的泛型方法3. Method类的泛型方法4. ParameterizedType类获取使用侧...
    99+
    2023-03-08
    Java获取泛型类型信息 Java 泛型类型信息
  • Java获取泛型类型信息
    根据使用泛型位置的不同可以分为:声明侧泛型、使用侧泛型。 声明侧的泛型信息被记录在Class文件的Constant pool中以Signature的形式保存。而使用侧的泛型信息并没有保存。 声明侧泛型 声明侧泛型包括: 泛型类,或泛型接口的...
    99+
    2023-09-18
    java 开发语言
  • Java中如何获取泛型类型信息
    这篇文章主要讲解了“Java中如何获取泛型类型信息”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Java中如何获取泛型类型信息”吧!根据使用泛型位置的不同可以分为:声明侧泛型、使用侧泛型。声...
    99+
    2023-07-05
  • Java 获取泛型的类型实例详解
    Java 获取泛型的类型实例详解Java 泛型实际上有很多缺陷,比如不能直接获取泛型的类型,不能获取带泛型类等。以下方式是不正确的:①.获取带泛型的类的类型Class lstUClazz = List<User>.class...
    99+
    2023-05-31
    java 泛型 类型
  • 在java中如何使用反射获取泛型信息
    ParameterizedType就是参数化类型的意思。解释:声明类型中带有“”的都是参数化类型,比如List,Map。getActualTypeArguments()返回Type[],即“”里的参数,比如Map。getRawType()返...
    99+
    2019-10-08
    java教程 java 反射 泛型信息
  • python泛型如何获取传入的类型
    Python中的泛型并不是强类型,因此无法直接获取传入的类型。 Python的类型注解主要是用来提供静态类型检查的,而在运行时并没有...
    99+
    2023-10-27
    python
  • 一篇文章带你了解java泛型--泛型类,泛型方法,泛型接口
    目录Java 泛型Java 泛型是什么泛型类泛型类的格式泛型类举例泛型方法泛型方法的格式泛型方法举例泛型接口泛型接口的格式泛型接口举例泛型接口实现类:测试接口类:打印结果:总结Jav...
    99+
    2024-04-02
  • go语言如何获取类型信息
    这篇文章主要介绍“go语言如何获取类型信息”,在日常操作中,相信很多人在go语言如何获取类型信息问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”go语言如何获取类型信息”的疑惑有所帮助!接下来,请跟着小编一起来...
    99+
    2023-07-05
  • go语言怎么获取类型信息
    在go语言中,可以通过反射来获取类型信息;只需要调用reflect包的TypeOf()函数即可。方法:1、使用“reflect.TypeOf()”函数取得指定变量的类型对象;2、使用Name()和Kind()获取类型对象的类型信息,语法“类...
    99+
    2023-05-14
    go语言 Golang
  • 获取不带类型参数的泛型结构的类型名称
    php小编草莓在这篇文章中将向大家介绍如何获取不带类型参数的泛型结构的类型名称。泛型是一种强大的编程技术,可以在不指定具体类型的情况下编写通用的代码。然而,有时我们可能需要获取泛型结构...
    99+
    2024-02-13
    typedef overflow
  • 详解java 中泛型中的类型擦除和桥方法
    在Java中,泛型的引入是为了在编译时提供强类型检查和支持泛型编程。为了实现泛型,Java编译器应用类型擦除实现:       1、  用类型参数(type parame...
    99+
    2023-05-31
    java 泛型 桥方法
  • java反射之获取类的信息方法(推荐)
    本文接上文“老生常谈反射之Class类的使用(必看篇)”,以编写一个用来获取类的信息(成员函数、成员变量、构造函数)的工具类来讲解"反射之获取类的信息"1、获取成员函数信息public static void printClassMetho...
    99+
    2023-05-31
    java 反射 获取类
  • Laravel如何获取模型信息
    本篇内容主要讲解“Laravel如何获取模型信息”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Laravel如何获取模型信息”吧!Laravel 模型信息 是 Spatie 的一个包,用于获取有...
    99+
    2023-07-04
  • 详解Java泛型中类型擦除问题的解决方法
    以前就了解过Java泛型的实现是不完整的,最近在做一些代码重构的时候遇到一些Java泛型类型擦除的问题,简单的来说,Java泛型中所指定的类型在编译时会将其去除,因此List&nbs...
    99+
    2024-04-02
  • Java中怎么擦除泛型类型
    Java中怎么擦除泛型类型,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。前言Java泛型这个特性是从JDK 1.5才开始加入的,因此为了兼容之前的版本,Java泛型的实现采...
    99+
    2023-06-20
  • JavaScala泛型(泛型方法,泛型类,泛型特质,上下界,协变、逆变、非变)
    目录1. 泛型1.1 泛型方法1.2 泛型类1.3 泛型特质2. 上下界2.1 上界2.2 下界3. 协变、逆变、非变3.1 非变3.3 逆变3.4 示例1. 泛型 泛型的意思是泛指...
    99+
    2023-05-14
    Scala泛型 Scala协变 Scala逆变
  • Java 泛型详解(超详细的java泛型方法解析)
    目录2. 什么是泛型3. 使用泛型的好处4. 泛型的使用4.1 泛型类4.2 泛型方法4.3 泛型接口5. 泛型通配符5.1 通配符基本使用5.2 通配符高级使用6. 总结1. 为什...
    99+
    2024-04-02
  • 在泛型 go 中模拟方法类型参数
    学习Golang要努力,但是不要急!今天的这篇文章《在泛型 go 中模拟方法类型参数》将会介绍到等等知识点,如果你想深入学习Golang,可以关注我!我会持续更新相关文章的,希望对大家都能有所帮助!...
    99+
    2024-04-04
  • 如何在不进行类型检查的情况下获取泛型函数中类型的大小?
    积累知识,胜过积蓄金银!毕竟在Golang开发的过程中,会遇到各种各样的问题,往往都是一些细节知识点还没有掌握好而导致的,因此基础知识点的积累是很重要的。下面本文《如何在不进行类型检查的情况下获取泛...
    99+
    2024-04-05
  • Java中怎么获取文件类型
    Java中怎么获取文件类型,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。1.根据文件扩展名判断优点:速度快,代码简单缺点:无法判断出真实的文件类型,例如一些伪造的文件或者没有后...
    99+
    2023-06-15
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作