美文网首页Android技术知识
java在运行时获取泛型类型

java在运行时获取泛型类型

作者: 会上网的井底之蛙 | 来源:发表于2017-05-27 14:42 被阅读2392次

    一个简单场景:一个处理函数,需要传入处理结果监听来响应结果,响应结果是一个JSON字符串,且要将其转换成对象再交给监听进行处理,由于对象类型存在多种,所以需要采用泛型。

    这样会存在一个需求,即将响应结果转换成相应的类型。为了达到该需求,我们一般会再传入一个Class来协助进行类型转换

    结果响应监听器:

    public interface IProcessResponse<T> {
    public void onProcessCompleted(T result);
    }

    处理函数:

    public class Test {

    public <T> void process(Class<T> reponseClass
    ,IProcessResponse<T> processResponseListener){
    String content = runProcess();

    Gson gson = new Gson();
    T responese = gson.fromJson(content, reponseClass);
    processResponseListener.onProcessCompleted(responese);
    }
    }

    这样处理很好,没有什么问题,但是我有一个疑问,既然监听器里已经带了T,它的类型实际上就是参数reponseClass参数所表示的,为什么还要传一个

    Class<T> reponseClass

    这个参数,难道不能直接获取到T所表示的Class吗?

    那么就引入了本篇要研究的主题,即如何在运行时获取泛型的类型。

    最直接的尝试:使用T.getClass()或者T.class。尝试后发现T并没有相应的接口函数供使用

    那就只能从实际的对象着手,即通过变量processResponseListener来取,因为传入的是实际存在的对象,可以通过反射获取泛型类型,改造一下process函数:

    public <T> void process(IProcessResponse<T> processResponseListener){
    String content = runProcess();

    Gson gson = new Gson();

    Type[] types = processResponseListener.getClass().getGenericInterfaces();
    Type[] params = ((ParameterizedType) types[0]).getActualTypeArguments();
    Class<T> reponseClass = (Class) params[0];


    T responese = gson.fromJson(content, reponseClass);

    processResponseListener.onProcessCompleted(responese);
    }

    getClass():获取的是实际运行的类的字节码

    getGenericInterfaces():以Type数组的形式返回本类直接实现的接口列表,包含了泛型参数信息

    getActualTypeArguments() :获取泛型类型的实际类型参数集

    另一种情况:

    假设我们的监听器不是一个接口,而是一个抽象类:

    public abstract class AbstractProcessResponse<T> {
    public abstract void onProcessCompleted(T result);
    private void commonProcess(){}
    }

    处理函数直接仿造 :

    public <T> void process(IProcessResponse<T> processResponseListener)

    函数,得到:

    public <T> void process(AbstractProcessResponse<T> processResponseListener){
    String content = runProcess();

    Gson gson = new Gson();

    Type[] types = processResponseListener.getClass().getGenericInterfaces();
    Type[] params = ((ParameterizedType) types[0]).getActualTypeArguments();
    Class<T> reponseClass = (Class) params[0];


    T responese = gson.fromJson(content, reponseClass);

    processResponseListener.onProcessCompleted(responese);
    }

    上面能得到正确结果吗?看getGenericInterfaces()方法的作用:

         以Type数组的形式返回本类直接实现的接口列表,包含了泛型参数信息

    很明显传入的AbstractProcessResponse<T>类型参数将是一个继承该抽象类的子类,所以getGenericInterfaces()取到的Type[]数组元素个数为0,上面的函数执行将会抛出异常

       经改造,正确的函数如下:

    public <T> void process(AbstractProcessResponse<T> processResponseListener){
    String content = runProcess();

    Gson gson = new Gson();

    Type type = processResponseListener.getClass().getGenericSuperclass();
    Type[] params = ((ParameterizedType) type).getActualTypeArguments();
    Class<T> reponseClass = (Class) params[0];

    T responese = gson.fromJson(content, reponseClass);

    processResponseListener.onProcessCompleted(responese);
    }

    getGenericSuperclass() :返回表示当前Class 所表示的实体(类、接口、基本类型或 void)的直接超类的Type

    最后总结一下:

    对于实现接口而来的对象,使用getGenericInterfaces()与getActualTypeArguments()

    对于继承父类而来的对象,使用getGenericSuperclass()与getActualTypeArguments()

    相关文章

      网友评论

      • 华枯荣:写得好,第一个把getGenericSuperclass和getGenericInterfaces说清楚

      本文标题:java在运行时获取泛型类型

      本文链接:https://www.haomeiwen.com/subject/rdtqfxtx.html