Class <T> entityClass = (Class <T>) ((ParameterizedType) getClass().getGenericSuperclass()).getActualTypeArguments()[0];
getGenericInterfaces()和getGenericSuperclass()
getInterfaces()和getSuperclass()
先來看看這兩個方法都是干什么用的:
1. public Type getGenericSuperclass()
用來返回表示當前Class 所表示的實體(類、接口、基本類型或 void)的直接超類的Type。如果這個直接超類是參數化類型的,則返回的Type對象必須明確反映在源代碼中聲明時使用的類型。比如:
import java.lang.reflect.ParameterizedType; public class GT1 extends GT2<Integer>{ public static void main(String[] args) { System.out.println(((ParameterizedType)new GT1().getClass().getGenericSuperclass())); } }
則輸出結果即為:
GT2<java.lang.Integer>
如果此Class代表的是Object 類、接口、基本類型或 void,則返回 null。。如果此對象表示一個數組類,則返回表示 Object 類的 Class 對象。
2. public Type[] getGenericInterfaces()
與上面那個方法類似,只不過Java的類可以實現多個接口,所以返回的Type必須用數組來存儲。
以上兩個方法返回的都是Type對象或數組,在我們的這個話題中,Class都是代表的參數化類型,因此可以將Type對象Cast成ParameterizedType對象。而 ParameterizedType對象有一個方法, getActualTypeArguments()。
public Type[] getActualTypeArguments()
用來返回一個Type對象數組,這個數組代表着這個Type聲明中實際使用的類型。接着使用上面的例子:
import java.lang.reflect.ParameterizedType; public class GT1 extends GT2<Integer>{ public static void main(String[] args) { System.out.println(((ParameterizedType)new GT1().getClass().getGenericSuperclass()).getActualTypeArguments()[0]); } }
這次的顯示結果將是:class java.lang.Integer
因此,我們可以通過繼承+反射的方法,來的到T.class。
需要說明的是,江南白衣使用的方法是將關鍵語句
Class < T > entityClass = (Class < T > ) ((ParameterizedType) getClass().getGenericSuperclass()).getActualTypeArguments()[ 0 ];
放在了超類,也就是聲明泛型的那個類的構造方法中。這樣一來,子類在繼承具有泛型的超類時,會自動調用超類的構造方法。在此超類的構造方法中,調用的getClass返回的是子類的Class類型(與通常的重寫機制有悖,呵呵,有待深究,但測試結果確是如此),則在子類中就無需再顯式地使用getGenericInterfaces()和getGenericSuperclass()等方法了。
接着,再使用(Class<T>)對 getActualTypeArguments()返回的元素做casting,即可得到所謂的T.class。