一个简单场景:一个处理函数,需要传入处理结果监听来响应结果,响应结果是一个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()