获取泛型参数实际类型的三种方式

通过反射可以获取类的字段和方法,但是有些类型是带有泛型参数的,那么进一步如何获取泛型的实际类型呢,常见的有三种方式,每一种方式都对应一种适应场景,不能混用:

  1. 获取类(接口)上的泛型参数的实际类型
  2. 获取方法参数上泛型参数的实际类型
  3. 获取字段上泛型参数的实际类型

代码如下:
定义一个接口和其实现类(测试验证用):

/**
 * 定义一个带泛型参数的接口
 */
public interface IMessageSender<T,V> {
    V sendMsg(T msg);
}

/**
 * 实现两接口:
 * 一个带泛型参数(验证获取其泛型参数),一个不带泛型参数(测试对比用)
 */
public class SmsMessageImpl implements 
        IMessageSender<String, List<Integer>>, Serializable {
    @Override
    public List<Integer> sendMsg(String msg) {
        return new ArrayList<>();
    }
}

获取泛型的实际类型的三种方式的代码示例:

public class TypeTest {

    /**
     * 1.获取类(接口)上的泛型参数的实际类型
     */
    @Test
    public void testClassGenericActualType(){
        //该类实现了两个接口分别是:IMessageSender<String, List<Integer>> 和 Serializable
        Type genericInterface1 = SmsMessageImpl.class.getGenericInterfaces()[0];
        Type genericInterface2 = SmsMessageImpl.class.getGenericInterfaces()[1];

        //第一个是带泛型参数的接口:IMessageSender<String, List<Integer>>,
        //因此Type实际上是ParameterizedType类型:sun.reflect.generics.reflectiveObjects.ParameterizedTypeImpl
        System.out.println(genericInterface1.getClass());

        //第二个是不带泛型参数的接口,因此Type是普通Class类型:class java.lang.Class
        System.out.println(genericInterface2.getClass());

        //获取泛化的接口参数的实际类型(先强转为ParameterizedType)
        Type[] actualTypeArguments = ((ParameterizedType) genericInterface1).getActualTypeArguments();
        //分别获取该类上的两个泛型参数的实际类型:String 和 List<Integer>
        System.out.println(actualTypeArguments[0] + " , "+ actualTypeArguments[1]);
        //其中List<Integer>可以继续获取其元素泛型的类型:Integer
        System.out.println(((ParameterizedType)actualTypeArguments[1]).getActualTypeArguments()[0]);

    }


    /**
     * 2. 获取方法参数上泛型参数的实际类型
     */
    @Test
    public void testMethodGenericActualType() throws NoSuchMethodException {
        Method method = TypeTest.class.getMethod("verify", Map.class);
        Parameter parameter = method.getParameters()[0];
        Type type = parameter.getParameterizedType();

        //由于参数是带泛型的,因此type的实际类型是ParameterizedType,
        //可以强制转换为ParameterizedType,然后获取参数Map中的两个泛型参数: <String,Integer>
        Type[] actualTypeArguments = ((ParameterizedType) type).getActualTypeArguments();
        System.out.println(actualTypeArguments[0] + " , " + actualTypeArguments[1]);

    }

    /**
     * 3.获取字段上泛型参数的实际类型
     */
    @Test
    public void testFieldGenericActualType() throws NoSuchFieldException {
        Field field = TypeTest.class.getDeclaredField("list");
        Type genericType = field.getGenericType();
        Type[] actualTypeArguments = ((ParameterizedType) genericType).getActualTypeArguments();
        //得到字段List<String>泛型中的实际类型String
        System.out.println(actualTypeArguments[0]);
    }


    public void verify(Map<String,Integer> map){
        //测试备用方法
    }
    
    private List<String> list;
    
}






  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值