java在泛型中得到T.class

來源:互聯網
上載者:User

 

泛型要知道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。

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.