您好,登錄后才能下訂單哦!
利用Java泛型怎么獲取T.class?很多新手對(duì)此不是很清楚,為了幫助大家解決這個(gè)難題,下面小編將為大家詳細(xì)講解,有這方面需求的人可以來(lái)學(xué)習(xí)下,希望你能有所收獲。
泛型的四點(diǎn)作用:
第一是泛化,可以拿個(gè)T代表任意類型。 但GP是被C++嚴(yán)苛的靜態(tài)性逼出來(lái)的,落到Java、C#這樣的花語(yǔ)平原里----所有對(duì)象除幾個(gè)原始類型外都派生于Object,再加上Java的反射功能,Java的Collection庫(kù)沒(méi)有范型一樣過(guò)得好好的。
第二是泛型 + 反射,原本因?yàn)镴ava的泛型拿不到T.class而覺(jué)得泛型沒(méi)用,最近才剛剛學(xué)到通過(guò)反射的API來(lái)獲取T的Class,后述。
第三是收斂,就是增加了類型安全,減少了強(qiáng)制類型轉(zhuǎn)換的代碼。這點(diǎn)倒是Java Collection歷來(lái)的弱項(xiàng)。
第四是可以在編譯期搞很多東西,比如MetaProgramming。但除非能完全封閉于框架內(nèi)部,框架的使用者和擴(kuò)展者都不用學(xué)習(xí)這些東西的用法,否則那就是自絕于人民的票房毒藥。C++的MetaProgramming好厲害吧,但對(duì)比一下Python拿Meta Programming生造一個(gè)Class出來(lái)的簡(jiǎn)便語(yǔ)法,就明白什么才是真正的叫好又叫座。
所以,作為一個(gè)架構(gòu)設(shè)計(jì)師,應(yīng)該使用上述的第2,3項(xiàng)用法,在框架類里配合使用反射和泛型,使得框架的能力更強(qiáng); 同時(shí)采用收斂特性,本著對(duì)人民負(fù)責(zé)的精神,用泛型使框架更加類型安全,更少?gòu)?qiáng)制類型轉(zhuǎn)換。
擦拭法避免了Java的流血分裂 :
大家經(jīng)常罵Java GP的擦拭法實(shí)現(xiàn),但我覺(jué)得多虧于它的中庸特性---如果你用就是范型,不用就是普通Object,避免了Java陣營(yíng)又要經(jīng)歷一場(chǎng)to be or not to be的分裂。
最大的例子莫過(guò)Java 5的Collection 框架, 比如有些同學(xué)堅(jiān)持認(rèn)為自己不會(huì)白癡到類型出錯(cuò),而且難以忍受每個(gè)定義的地方都要帶一個(gè)泛型定義List〈Book〉,不用強(qiáng)制類型轉(zhuǎn)換所省下的代碼還不夠N處定義花的(對(duì)了,java里面還沒(méi)有tyepdef.....),因此對(duì)范型十分不感冒,這時(shí)就要齊齊感謝這個(gè)搽拭法讓你依然可以對(duì)一個(gè)泛型框架保持非泛型的用法了...
<<干貨來(lái)了!!!!!!>>
通過(guò)反射獲得 T.class:
abstract public class BaseHibernateEntityDao<T> extends HibernateDaoSupport { private Class<T> entityClass; public BaseHibernateEntityDao() { entityClass =(Class<T>) ((ParameterizedType) getClass() .getGenericSuperclass()).getActualTypeArguments()[0]; } public T get(Serializable id) { T o = (T) getHibernateTemplate().get(entityClass, id); return o; } }
重點(diǎn)就是這句話:
Class<T> entityClass = (Class<T>) ((ParameterizedType) getClass().getGenericSuperclass()).getActualTypeArguments()[0];
解釋:
1. public Type getGenericSuperclass()
用來(lái)返回表示當(dāng)前Class 所表示的實(shí)體(類、接口、基本類型或 void)的直接超類的Type。如果這個(gè)直接超類是參數(shù)化類型的,則返回的Type對(duì)象必須明確反映在源代碼中聲明時(shí)使用的類型。比如:
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())); } }
則輸出結(jié)果即為:
GT2<java.lang.Integer>
如果此Class代表的是Object 類、接口、基本類型或 void,則返回 null。。如果此對(duì)象表示一個(gè)數(shù)組類,則返回表示 Object 類的 Class 對(duì)象。
2. public Type[] getGenericInterfaces()
與上面那個(gè)方法類似,只不過(guò)Java的類可以實(shí)現(xiàn)多個(gè)接口,所以返回的Type必須用數(shù)組來(lái)存儲(chǔ)。
以上兩個(gè)方法返回的都是Type對(duì)象或數(shù)組,在我們的這個(gè)話題中,Class都是代表的參數(shù)化類型,因此可以將Type對(duì)象Cast成ParameterizedType對(duì)象。而 ParameterizedType對(duì)象有一個(gè)方法, getActualTypeArguments()。
public Type[] getActualTypeArguments()
用來(lái)返回一個(gè)Type對(duì)象數(shù)組,這個(gè)數(shù)組代表著這個(gè)Type聲明中實(shí)際使用的類型。接著使用上面的例子:
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 ];
放在了超類,也就是聲明泛型的那個(gè)類的構(gòu)造方法中。這樣一來(lái),子類在繼承具有泛型的超類時(shí),會(huì)自動(dòng)調(diào)用超類的構(gòu)造方法。在此超類的構(gòu)造方法中,調(diào)用的getClass返回的是子類的Class類型(與通常的重寫(xiě)機(jī)制有悖,呵呵,有待深究,但測(cè)試結(jié)果確是如此),則在子類中就無(wú)需再顯式地使用getGenericInterfaces()和getGenericSuperclass()等方法了。
接著,再使用(Class<T>)對(duì) getActualTypeArguments()返回的元素做casting,即可得到所謂的T.class。
泛型之后,所有BaseHibernateEntityDao的子類只要定義了泛型,就無(wú)需再重載getEnttityClass(),get()函數(shù)和find()函數(shù),銷益挺明顯的,所以SpringSide的Dao基類毫不猶豫就泛型了。
不過(guò)擦拭法的大棒仍在,所以子類的泛型語(yǔ)法可不能亂寫(xiě),最正確的用法只有:
public class BookDao extends BaseHibernateEntityDao<Book>
看完上述內(nèi)容是否對(duì)您有幫助呢?如果還想對(duì)相關(guān)知識(shí)有進(jìn)一步的了解或閱讀更多相關(guān)文章,請(qǐng)關(guān)注億速云行業(yè)資訊頻道,感謝您對(duì)億速云的支持。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。