一、场景说明:
我们在使用GSON自动解析json字符串的时候,需要传参一个Class参数,告诉它需要解析成什么对象,而这一步骤我们常常通过底层自动化框架+泛型来实现,这时就会涉及到读取泛型类型的问题。
二、目前网上说的方法,大都是这种:
Class<T> targetClass = (Class<T>)((ParameterizedType)XXX.getClass().getGenericInterfaces()[0]).getActualTypeArguments()[0];
这段代码的含义:
获取此类的第一个实现接口的第一个泛型参数类型Type,并强转成Class。
显然,这个方法存在几个致命的缺陷:
1、对实现类中implements接口的顺序敏感,如果要寻找的接口不是该类第一个实现的接口,那就会取到错误的泛型类型。
1、如果要寻找的接口实现在父类中,则无法找到,因为getGenericInterfaces()只会获取当前类的实现接口列表,不会获取父类的。
3、强转Class,意味着如果泛型类型不是Class类型,程序就会崩溃。(泛型中有可能嵌套了一层泛型,比如MyInterface<ArrayList<Serializable>>, 这种情况下对应的是ParameterizedType类型,并不是Class。)
*(注:如果要查找一个抽象类的泛型类型,则把getGenericInterfaces()换成getGenericSuperclass()方法即可,但也同样存在上述的第2、3问题)
三、思考如何改进:
对于问题1:能解决。通过名称遍历所有实现的接口,直到找到对应的接口。
对于问题2:能解决。写个递归,通过Class.getSuperclass()方法去检查父类中实现的接口。
对于问题3:不能解决。对于不是Class类型的Type,虽然能够获取到它的类型名称,但是也无法通过Class.forName()方法转换成Class类型,会报“ClassNotFoundException”异常。所以对于这种泛型类型,唯一的用处是可以获取到它的完整名称。 (但对于上述使用场景没有太大影响,因为需要用JSON解析的对象,一般不会是一个带多重泛型的类。若真是遇到带多重泛型的类,则可以写一个新类去继承它 以解决此问题了)
四、改进后的靠谱方案:
(由于代码写完后暂时还没测试,为避免误导读者,待测试完全通过后补充记录~ 急的朋友可以根据我上面写的思路自己试一试。)