Java通过Class的对象来获取泛型的class示例
轉(zhuǎn)載自??Java通過(guò)Class的對(duì)象來(lái)獲取泛型的class示例
在使用spring的JdbcTemplate實(shí)現(xiàn)DAO的時(shí)候,經(jīng)常會(huì)用到一個(gè)類(lèi)ParameterizedBeanPropertyRowMapper。它的靜態(tài)方法newInstance()接受一個(gè)Class類(lèi)型的參數(shù),用于將ResultSet中的屬性映射到傳入的這個(gè)Class類(lèi)型的Bean對(duì)象中,再組成列表返回。
?
如果要想把這個(gè)DAO做成泛型的,就必須要知道Class的類(lèi)型。但是直接寫(xiě)成T.class顯然是不行的。從網(wǎng)上查了不少資料,結(jié)果只有一個(gè),由于Java的泛型實(shí)現(xiàn)使用了“擦拭法”(具體細(xì)節(jié)沒(méi)深究,呵呵),導(dǎo)致Java的泛型不能直接獲取到自身的聲明的泛型類(lèi)型。
?
不過(guò)從江南白衣的blog文章里搜到了有用的東西:使用反射來(lái)獲得“T.class”。
原文地址:http://www.blogjava.net/calvin/archive/2009/12/10/43830.html
?
主要用到的是這么一句:
[c-sharp] view plain copy
Class <T> entityClass = (Class <T>) ((ParameterizedType) getClass().getGenericSuperclass()).getActualTypeArguments()[0]; ??
?
我查詢(xún)了Java API,在Class類(lèi)中有這么兩個(gè)方法: getGenericInterfaces()和getGenericSuperclass()
先來(lái)看看這兩個(gè)方法都是干什么用的:
1. public Type getGenericSuperclass()
用來(lái)返回表示當(dāng)前Class 所表示的實(shí)體(類(lèi)、接口、基本類(lèi)型或 void)的直接超類(lèi)的Type。如果這個(gè)直接超類(lèi)是參數(shù)化類(lèi)型的,則返回的Type對(duì)象必須明確反映在源代碼中聲明時(shí)使用的類(lèi)型。比如:
?
如果此Class代表的是Object 類(lèi)、接口、基本類(lèi)型或 void,則返回 null。。如果此對(duì)象表示一個(gè)數(shù)組類(lèi),則返回表示 Object 類(lèi)的 Class 對(duì)象。
?
2. public Type[] getGenericInterfaces()
與上面那個(gè)方法類(lèi)似,只不過(guò)Java的類(lèi)可以實(shí)現(xiàn)多個(gè)接口,所以返回的Type必須用數(shù)組來(lái)存儲(chǔ)。
?
以上兩個(gè)方法返回的都是Type對(duì)象或數(shù)組,在我們的這個(gè)話(huà)題中,Class都是代表的參數(shù)化類(lèi)型,因此可以將Type對(duì)象Cast成ParameterizedType對(duì)象。而 ParameterizedType對(duì)象有一個(gè)方法, getActualTypeArguments()。
public Type[] getActualTypeArguments()
用來(lái)返回一個(gè)Type對(duì)象數(shù)組,這個(gè)數(shù)組代表著這個(gè)Type聲明中實(shí)際使用的類(lèi)型。接著使用上面的例子:
package com.mot.hyena.test; 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]); } } 這次的顯示結(jié)果將是:class java.lang.Integer?
因此,我們可以通過(guò)繼承+反射的方法,來(lái)的到T.class。
?
需要說(shuō)明的是,江南白衣使用的方法是將關(guān)鍵語(yǔ)句
Class < T > ?entityClass ?= ?(Class < T > ) ((ParameterizedType) getClass().getGenericSuperclass()).getActualTypeArguments()[ 0 ];
放在了超類(lèi),也就是聲明泛型的那個(gè)類(lèi)的構(gòu)造方法中。這樣一來(lái),子類(lèi)在繼承具有泛型的超類(lèi)時(shí),會(huì)自動(dòng)調(diào)用超類(lèi)的構(gòu)造方法。在此超類(lèi)的構(gòu)造方法中,調(diào)用的getClass返回的是子類(lèi)的Class類(lèi)型(與通常的重寫(xiě)機(jī)制有悖,呵呵,有待深究,但測(cè)試結(jié)果確是如此),則在子類(lèi)中就無(wú)需再顯式地使用 getGenericInterfaces()和getGenericSuperclass()等方法了。
?
接著,再使用(Class<T>)對(duì) getActualTypeArguments()返回的元素做casting,即可得到所謂的T.class。
完整示例:
Java通過(guò)Class的對(duì)象來(lái)獲取泛型的class示例
[java]?view plaincopy此方法一般在模板方法中經(jīng)常用到,因此做一個(gè)筆記!
總結(jié)
以上是生活随笔為你收集整理的Java通过Class的对象来获取泛型的class示例的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: java反射的field.get(nul
- 下一篇: java美元兑换,(Java实现) 美元