美文网首页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()

相关文章

  • Java如何在运行时获取泛型的类型

    Java泛型是伪泛型,会在编译完成时进行类型的擦除,我们无法在运行时获取泛型参数的具体类型(类型擦除会被替换成泛型...

  • Java泛型的原始类型

    众所周知,Java中的泛型在编译期被擦除,那有没有办法在运行时获取到泛型的原始类型呢?有的。 获取泛型类型 如果定...

  • java在运行时获取泛型类型

    一个简单场景:一个处理函数,需要传入处理结果监听来响应结果,响应结果是一个JSON字符串,且要将其转换成对象再交给...

  • JAVA泛型和类型擦除

    什么是类型擦除 Java是使用擦除来实现泛型的。使用泛型后在运行时任何具体的类型信息都被擦除了,关于泛型的处理都是...

  • Java 中的泛型

    泛型的目的:Java 泛型就是把一种语法糖,通过泛型使得在编译阶段完成一些类型转换的工作,避免在运行时强制类型转换...

  • 【Java】【反射】泛型反射

    泛型反射 在运行时,泛型是无效的,所以可以通过反射在运行时将其他类型变量添加到集合,而不需要考虑泛型

  • Java泛型实现机制

    原理Java泛型是通过类型擦除来实现。 类型擦除的好处运行时内存负担小诸如List 、List 等在运行时只会存在...

  • 03-Dart语法--对象的构建者以及操控者

    typedef 泛型 Dart 的泛型类型是固化的,在运行时有也可以判断具体的类型。 控制语句 if and el...

  • 什么情况下不能使用 Java 泛型

    1. 前言 Java 1.5 引入了泛型来保证类型安全,防止在运行时发生类型转换异常,让类型参数化,提高了代码的可...

  • 泛型

    强制转换必须知道类型,否则就会在运行时报错 在java中增加泛型之前,泛型程序设计使用继承来实现的 坏处:需要强制...

网友评论

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

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

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