假如你定義了一個有泛型的Base類,如下BaseClass<DO, VO>代碼示例,如何在BaseClass中獲取到子類對應(yīng)的DO、VO的class對象呢?
class BaseClass<DO, VO> {
Class<DO> doClass;
Class<VO> voClass;
...
}
1. 傳統(tǒng)方式:通過參數(shù)傳遞或由子類指定
//通過參數(shù)傳遞
class BaseClass<DO, VO> {
//Class<?>定義為參數(shù),通過參數(shù)傳遞
public myMethod(..., Class<?> clazz){
...
}
}
//由子類指定
class UserClass extends BaseClass<UserDO, UserVO> {
...{
super.doClass = UserDO.class;
super.voClass = UserVO.class;
}
}
這種方式非常簡單容易理解,但會在方法/子類中產(chǎn)生冗余代碼。
下面還有兩種更優(yōu)雅的實現(xiàn)代碼:
2. 在Base類中通過Java反射統(tǒng)一獲取
class BaseClass<DO, VO> {
...{
Type[] types = ((ParameterizedType)getClass().getGenericSuperclass()).getActualTypeArguments();
this.doClass = (Class<DO>)types[0];
this.voClasss = (Class<VO>)types[1];
}
}
// 如果是接口獲取實現(xiàn)類的泛型class,需將getGenericSuperclass()替換成getGenericInterfaces()
3. 在Base類中通過Spring統(tǒng)一獲取
class BaseClass<DO, VO> {
...{
ResolvableType resolvableType = ResolvableType.forClass(this.getClass()).getSuperType();
ResolvableType[] types = resolvableType.getSuperType().getGenerics();
this.doClass = (Class<DO>) types[0].resolve();
this.voClasss = (Class<VO>) types[1].resolve();
}
}