Java反射获取泛型类型

Java反射获取泛型类型

              在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才可以。


Java 17中,可以使用反射获取类型。面是一种常见的方法: 1. 首先,通过反射获取目标类的Class对象。假设目标类为`MyClass`,可以使用`MyClass.class`或者`Class.forName("com.example.MyClass")`来获取。 2. 使用`getDeclaredField`方法获取目标字段的Field对象。假设目标字段为`myField`,可以使用`Class.getDeclaredField("myField")`来获取。 3. 通过Field对象的`getGenericType`方法获取字段的类型。这将返回一个Type对象,表示字段的实际类型。 4. 如果字段的类型是参数化类型(即包含参数),可以通过Type对象的一些方法来获取参数的信息。例如,可以使用`ParameterizedType`接口来获取参数化类型的原始类型参数列表。 下面是一个示例代码,演示了如何使用反射获取类型: ```java import java.lang.reflect.Field; import java.lang.reflect.ParameterizedType; import java.lang.reflect.Type; public class Main { public static void main(String[] args) throws NoSuchFieldException { // 获取目标类的Class对象 Class<MyClass> clazz = MyClass.class; // 获取目标字段的Field对象 Field field = clazz.getDeclaredField("myField"); // 获取字段的类型 Type fieldType = field.getGenericType(); // 如果字段的类型是参数化类型 if (fieldType instanceof ParameterizedType) { ParameterizedType parameterizedType = (ParameterizedType) fieldType; // 获取参数化类型的原始类型 Type rawType = parameterizedType.getRawType(); System.out.println("Raw type: " + rawType); // 获取参数化类型参数列表 Type[] typeArguments = parameterizedType.getActualTypeArguments(); for (Type typeArgument : typeArguments) { System.out.println("Type argument: " + typeArgument); } } } } class MyClass { private List<String> myField; } ``` 在上面的示例中,我们通过反射获取了`MyClass`类中名为`myField`的字段的类型。输出结果如下: ``` Raw type: interface java.util.List Type argument: class java.lang.String ``` 这表明`myField`字段的类型是`List<String>`,其中`List`是原始类型,`String`是参数。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值