泛型要知道Class的類型。但是直接寫成T.class顯然是不行的。從網上查了不少資料,結果只有一個,由於Java的泛型實現使用了“擦拭法”(具體細節沒深究,呵呵),導致Java的泛型不能直接擷取到自身的聲明的泛型型別。
不過從江南白衣的blog文章裡搜到了有用的東西:使用反射來獲得“T.class”。
原文地址:http://www.blogjava.net/calvin/archive/2009/12/10/43830.html
主要用到的是這麼一句:
Class <T> entityClass = (Class <T>) ((ParameterizedType) getClass().getGenericSuperclass()).getActualTypeArguments()[0];<br />
我查詢了Java API,在Class類中有這麼兩個方法: getGenericInterfaces()和getGenericSuperclass()
先來看看這兩個方法都是幹什麼用的:
1. public Type getGenericSuperclass()
用來返回表示當前Class 所表示的實體(類、介面、基本類型或 void)的直接超類的Type。如果這個直接超類是參數化型別的,則返回的Type對象必須明確反映在原始碼中聲明時使用的類型。比如:
package com.mot.hyena.test;<br />import java.lang.reflect.ParameterizedType;<br />public class GT1 extends GT2<Integer>{<br />public static void main(String[] args) {<br />System.out.println(((ParameterizedType)new GT1().getClass().getGenericSuperclass()));<br />}<br />}
則輸出結果即為:com.mot.hyena.test.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聲明中實際使用的類型。接著使用上面的例子:
package com.mot.hyena.test;<br />import java.lang.reflect.ParameterizedType;<br />public class GT1 extends GT2<Integer>{<br />public static void main(String[] args) {<br />System.out.println(((ParameterizedType)new GT1().getClass().getGenericSuperclass()).getActualTypeArguments()[0]);<br />}<br />}
這次的顯示結果將是: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。