Java获取泛型的Class的工具类

本文介绍了一个实用的Java工具类,用于通过反射机制获取指定类的父类泛型参数的具体类型。这对于处理泛型编程中类型擦除带来的问题非常有用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

package com.grape.jxc2.utils;


import java.lang.reflect.ParameterizedType;    
import java.lang.reflect.Type;    
    
public class GenericsUtils {    
    /**    
     * 通过反射,获得定义Class时声明的父类的范型参数的类型.    
     * 如public BookManager extends GenricManager<Book>    
     *    
     * @param clazz The class to introspect    
     * @return the first generic declaration, or <code>Object.class</code> if cannot be determined    
     */    
    @SuppressWarnings("rawtypes")
public static Class getSuperClassGenricType(Class clazz) {    
        return getSuperClassGenricType(clazz, 0);    
    }    
    
    /**    
     * 通过反射,获得定义Class时声明的父类的范型参数的类型.    
     * 如public BookManager extends GenricManager<Book>    
     *    
     * @param clazz clazz The class to introspect    
     * @param index the Index of the generic ddeclaration,start from 0.    
     */    
    @SuppressWarnings("rawtypes")
public static Class getSuperClassGenricType(Class clazz, int index) throws IndexOutOfBoundsException {    
    
        Type genType = clazz.getGenericSuperclass();    
    
        if (!(genType instanceof ParameterizedType)) {    
            return Object.class;    
        }    
    
        Type[] params = ((ParameterizedType) genType).getActualTypeArguments();    
    
        if (index >= params.length || index < 0) {    
            return Object.class;    
        }    
        if (!(params[index] instanceof Class)) {    
            return Object.class;    
        }    
        return (Class) params[index];    
    }    
}    
### 如何在Java获取类的信息 #### 获取`Class`对象的方法 为了处理并保持类安全,在运行时可以通过多种方式获取的`Class`实例。由于Java中的类擦除机制,直接通过普通的反射可能无法完全保留信息,但是仍然存在一些技巧来绕过这个问题。 对于带有具体类参数,可以直接使用`.class`语法得到其对应的`Class`对象,不过这仅适用于原始类而非实际的参数化类。例如: ```java // 这里只得到了List这个原始类Class对象, 而不是具体的 List<String> 类 Class<?> clazz = ArrayList.class; ``` 当需要表示特定参数化的类(如`ArrayList<String>`),则应考虑创建一个匿名内部类或利用其他工具库的帮助函数。一种常见做法是定义一个辅助方法接受类令牌作为参数[^3]。 下面是一个更复杂的例子,展示了如何捕获的实际类参数,并将其封装在一个自定义的类标记内以便后续操作: ```java public abstract class TypeReference<T> { private final Class<T> type; protected TypeReference() { ParameterizedType superclass = (ParameterizedType) getClass().getGenericSuperclass(); if (superclass == null || !(superclass instanceof ParameterizedType)) { throw new IllegalArgumentException("Internal error: TypeReference constructed without actual type information"); } this.type = (Class<T>) ((ParameterizedType) getClass().getGenericSuperclass()).getActualTypeArguments()[0]; } public Class<T> getType() { return type; } } ``` 此模式允许开发者编写如下代码片段以保存确切的签名: ```java new TypeReference<List<String>>() {}; ``` 上述表达式会返回一个新的未命名子类的对象,该子类继承自`TypeReference<List<String>>`,并且能够正确解析出`List<String>`的具体类信息。 另外,如果是在已知上下文中工作,则可以尝试访问字段、方法声明等处记录下来的元数据。比如给定某个包含属性的类,可通过反射API读取这些成员变量上的描述。 综上所述,虽然Java虚拟机执行过程中会发生类擦除现象,使得部分细节丢失;但借助于适当的设计模式和技术手段,还是能够在一定程度上恢复和利用这类信息。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值