泛型,一個(gè)孤獨(dú)的守門者。 大家可能會(huì)有疑問(wèn),我為什么叫做泛型是一個(gè)守門者。這其實(shí)是我個(gè)人的看法而已,我的意思是說(shuō)泛型沒(méi)有其看起來(lái)那么深不可測(cè),它并不神秘與神奇。泛型是 Java 中一個(gè)很小巧的概念,但同時(shí)也是一個(gè)很容易讓人迷惑的知識(shí)點(diǎn),它讓人迷惑的地方在于它的許多表現(xiàn)有點(diǎn)違反直覺(jué)。 文章開(kāi)始的地方,先給大家奉上一道經(jīng)典的測(cè)試題。 List 請(qǐng)問(wèn),上面代碼最終結(jié)果輸出的是什么?不了解泛型的和很熟悉泛型的同學(xué)應(yīng)該能夠答出來(lái),而對(duì)泛型有所了解,但是了解不深入的同學(xué)可能會(huì)答錯(cuò)。 正確答案是 true。 上面的代碼中涉及到了泛型,而輸出的結(jié)果緣由是類型擦除。先好好說(shuō)說(shuō)泛型。 泛型是什么?泛型的英文是 generics,generic 的意思是通用,而翻譯成中文,泛應(yīng)該意為廣泛,型是類型。所以泛型就是能廣泛適用的類型。 但泛型還有一種較為準(zhǔn)確的說(shuō)法就是為了參數(shù)化類型,或者說(shuō)可以將類型當(dāng)作參數(shù)傳遞給一個(gè)類或者是方法。 那么,如何解釋類型參數(shù)化呢? public class Cache { 假設(shè) Cache 能夠存取任何類型的值,于是,我們可以這樣使用它。 Cache cache = new Cache(); 使用的方法也很簡(jiǎn)單,只要我們做正確的強(qiáng)制轉(zhuǎn)換就好了。 但是,泛型卻給我們帶來(lái)了不一樣的編程體驗(yàn)。 public class Cache 這就是泛型,它將 value 這個(gè)屬性的類型也參數(shù)化了,這就是所謂的參數(shù)化類型。再看它的使用方法。 CacheString> cache1 = new CacheString>(); 最顯而易見(jiàn)的好處就是它不再需要對(duì)取出來(lái)的結(jié)果進(jìn)行強(qiáng)制轉(zhuǎn)換了。但,還有另外一點(diǎn)不同。 泛型除了可以將類型參數(shù)化外,而參數(shù)一旦確定好,如果類似不匹配,編譯器就不通過(guò)。 所以,綜合上面信息,我們可以得到下面的結(jié)論。
下面的文章,我們正常介紹泛型的相關(guān)知識(shí)。 泛型的定義和使用泛型按照使用情況可以分為 3 種。 泛型類我們可以這樣定義一個(gè)泛型類。 public class Test 尖括號(hào) public class Test 但出于規(guī)范的目的,Java 還是建議我們用單個(gè)大寫字母來(lái)代表類型參數(shù)。常見(jiàn)的如: 如果一個(gè)類被 那么對(duì)于泛型類怎么樣使用呢? TestString> test1 = new Test<>(); 只要在對(duì)泛型類創(chuàng)建實(shí)例的時(shí)候,在尖括號(hào)中賦值相應(yīng)的類型便是。T 就會(huì)被替換成對(duì)應(yīng)的類型,如 String 或者是 Integer。你可以相像一下,當(dāng)一個(gè)泛型類被創(chuàng)建時(shí),內(nèi)部自動(dòng)擴(kuò)展成下面的代碼。 public class Test 當(dāng)然,泛型類不至接受一個(gè)類型參數(shù),它還可以這樣接受多個(gè)類型參數(shù)。 public class MultiType 泛型方法public class Test1 { 泛型方法與泛型類稍有不同的地方是,類型參數(shù)也就是尖括號(hào)那一部分是寫在返回值前面的。 當(dāng)然,聲明的類型參數(shù),其實(shí)也是可以當(dāng)作返回值的類型的。 public 泛型類與泛型方法的共存現(xiàn)象public class Test1 上面代碼中, 所以,針對(duì)上面的代碼,我們可以這樣編寫測(cè)試代碼。 Test1String> t = new Test1(); 泛型類的實(shí)際類型參數(shù)是 String,而傳遞給泛型方法的類型參數(shù)是 Integer,兩者不想干。 但是,為了避免混淆,如果在一個(gè)泛型類中存在泛型方法,那么兩者的類型參數(shù)最好不要同名。比如, public class Test1 泛型接口泛型接口和泛型類差不多,所以一筆帶過(guò)。 public interface IterableT> { 通配符 ?除了用 可能有同學(xué)會(huì)想,已經(jīng)有了 class Base{} 上面代碼顯示,Base 是 Sub 的父類,它們之間是繼承關(guān)系,所以 Sub 的實(shí)例可以給一個(gè) Base 引用賦值,那么 List lsub = new ArrayList<>(); 最后一行代碼成立嗎?編譯會(huì)通過(guò)嗎? 答案是否定的。 編譯器不會(huì)讓它通過(guò)的。Sub 是 Base 的子類,不代表 但是,在現(xiàn)實(shí)編碼中,確實(shí)有這樣的需求,希望泛型能夠處理某一范圍內(nèi)的數(shù)據(jù)類型,比如某個(gè)類和它的子類,對(duì)此 Java 引入了通配符這個(gè)概念。 所以,通配符的出現(xiàn)是為了指定泛型中的類型范圍。 通配符有 3 種形式。
無(wú)限定通配符public void testWildCards(Collection collection){ 上面的代碼中,方法內(nèi)的參數(shù)是被無(wú)限定通配符修飾的 Collection 對(duì)象,它隱略地表達(dá)了一個(gè)意圖或者可以說(shuō)是限定,那就是 testWidlCards() 這個(gè)方法內(nèi)部無(wú)需關(guān)注 Collection 中的真實(shí)類型,因?yàn)樗俏粗?/span>。所以,你只能調(diào)用 Collection 中與類型無(wú)關(guān)的方法。 我們可以看到,當(dāng) List<>> wildlist = new ArrayList 有人說(shuō), 有同學(xué)可能會(huì)想, 個(gè)人認(rèn)為,提高了代碼的可讀性,程序員看到這段代碼時(shí),就能夠迅速對(duì)此建立極簡(jiǎn)潔的印象,能夠快速推斷源碼作者的意圖。
public void testSub(Collection para){ 上面代碼中,para 這個(gè) Collection 接受 Base 及 Base 的子類的類型。 但是,它仍然喪失了寫操作的能力。也就是說(shuō) para.add(new Sub()); 仍然編譯不通過(guò)。 沒(méi)有關(guān)系,我們不知道具體類型,但是我們至少清楚了類型的范圍。 這個(gè)和 public void testSuper(Collection<>super Sub> para){
public void testSuper(Collection para){ 通配符與類型參數(shù)的區(qū)別一般而言,通配符能干的事情都可以用類型參數(shù)替換。 public void testWildCards(Collection collection){} 可以被 public 取代。 值得注意的是,如果用泛型方法來(lái)取代通配符,那么上面代碼中 collection 是能夠進(jìn)行寫操作的。只不過(guò)要進(jìn)行強(qiáng)制轉(zhuǎn)換。 public 需要特別注意的是,類型參數(shù)適用于參數(shù)之間的類別依賴關(guān)系,舉例說(shuō)明。 public class Test2 T,E extends T>{ public E 類型是 T 類型的子類,顯然這種情況類型參數(shù)更適合。 public 如果一個(gè)方法的返回類型依賴于參數(shù)的類型,那么通配符也無(wú)能為力。 public T test1(T t){ 類型擦除泛型是 Java 1.5 版本才引進(jìn)的概念,在這之前是沒(méi)有泛型的概念的,但顯然,泛型代碼能夠很好地和之前版本的代碼很好地兼容。 這是因?yàn)椋?span>泛型信息只存在于代碼編譯階段,在進(jìn)入 JVM 之前,與泛型相關(guān)的信息會(huì)被擦除掉,專業(yè)術(shù)語(yǔ)叫做類型擦除。 通俗地講,泛型類和普通類在 java 虛擬機(jī)內(nèi)是沒(méi)有什么特別的地方?;仡櫸恼麻_(kāi)始時(shí)的那段代碼 List 打印的結(jié)果為 true 是因?yàn)?nbsp; 泛型信息被擦除了。 可能同學(xué)會(huì)問(wèn),那么類型 String 和 Integer 怎么辦? 答案是泛型轉(zhuǎn)譯。 public class Erasure Erasure 是一個(gè)泛型類,我們查看它在運(yùn)行時(shí)的狀態(tài)信息可以通過(guò)反射。 ErasureString> erasure = new ErasureString>('hello'); 打印的結(jié)果是 erasure class is:com.frank.test.Erasure Class 的類型仍然是 Erasure 并不是 Field[] fs = eclz.getDeclaredFields(); 打印結(jié)果是 Field name object type:java.lang.Object 那我們可不可以說(shuō),泛型類被類型擦除后,相應(yīng)的類型就被替換成 Object 類型呢? 這種說(shuō)法,不完全正確。 我們更改一下代碼。 public class Erasure T extends String>{ 現(xiàn)在再看測(cè)試結(jié)果: Field name object type:java.lang.String 我們現(xiàn)在可以下結(jié)論了,在泛型類被類型擦除的時(shí)候,之前泛型類中的類型參數(shù)部分如果沒(méi)有指定上限,如 所以,在反射中。 public class Erasure add() 這個(gè)方法對(duì)應(yīng)的 Method 的簽名應(yīng)該是 Object.class。 Erasure 打印結(jié)果是 method:public void com.frank.test.Erasure.add(java.lang.Object) 也就是說(shuō),如果你要在反射中找到 add 對(duì)應(yīng)的 Method,你應(yīng)該調(diào)用 類型擦除帶來(lái)的局限性類型擦除,是泛型能夠與之前的 java 版本代碼兼容共存的原因。但也因?yàn)轭愋筒脸?,它?huì)抹掉很多繼承相關(guān)的特性,這是它帶來(lái)的局限性。 理解類型擦除有利于我們繞過(guò)開(kāi)發(fā)當(dāng)中可能遇到的雷區(qū),同樣理解類型擦除也能讓我們繞過(guò)泛型本身的一些限制。比如 正常情況下,因?yàn)榉盒偷南拗疲幾g器不讓最后一行代碼編譯通過(guò),因?yàn)轭愃撇黄ヅ洌?,基于?duì)類型擦除的了解,利用反射,我們可以繞過(guò)這個(gè)限制。 public interface ListE> extends CollectionE>{ 上面是 List 和其中的 add() 方法的源碼定義。 因?yàn)?E 代表任意的類型,所以類型擦除時(shí),add 方法其實(shí)等同于 boolean add(Object obj); 那么,利用反射,我們繞過(guò)編譯器去調(diào)用 add 方法。 public class ToolTest { 打印結(jié)果是: 23 可以看到,利用類型擦除的原理,用反射的手段就繞過(guò)了正常開(kāi)發(fā)中編譯器不允許的操作限制。 泛型中值得注意的地方泛型類或者泛型方法中,不接受 8 種基本數(shù)據(jù)類型。所以,你沒(méi)有辦法進(jìn)行這樣的編碼。 List 需要使用它們對(duì)應(yīng)的包裝類。 List 對(duì)泛型方法的困惑public 有的同學(xué)可能對(duì)于連續(xù)的兩個(gè) T 感到困惑,其實(shí) test('123'); 那么實(shí)際上相當(dāng)于 public String test(String t); Java 不能創(chuàng)建具體類型的泛型數(shù)組這句話可能難以理解,代碼說(shuō)明。 List 這兩行代碼是無(wú)法在編譯器中編譯通過(guò)的。原因還是類型擦除帶來(lái)的影響。
但是, List<>>[] li3 = new ArrayList<>>[10]; 借助于無(wú)限定通配符卻可以,前面講過(guò) 泛型,并不神奇我們可以看到,泛型其實(shí)并沒(méi)有什么神奇的地方,泛型代碼能做的非泛型代碼也能做。 而類型擦除,是泛型能夠與之前的 java 版本代碼兼容共存的原因。 可量也正因?yàn)轭愋筒脸龑?dǎo)致了一些隱患與局限。 但,我還是要建議大家使用泛型,如官方文檔所說(shuō)的,如果可以使用泛型的地方,盡量使用泛型。 畢竟它抽離了數(shù)據(jù)類型與代碼邏輯,本意是提高程序代碼的簡(jiǎn)潔性和可讀性,并提供可能的編譯時(shí)類型轉(zhuǎn)換安全檢測(cè)功能。 類型擦除不是泛型的全部,但是它卻能很好地檢測(cè)我們對(duì)于泛型這個(gè)概念的理解程度。 我在文章開(kāi)頭將泛型比作是一個(gè)守門人,原因就是他本意是好的,守護(hù)我們的代碼安全,然后在門牌上寫著出入的各項(xiàng)規(guī)定,及“xxx 禁止出入”的提醒。但是同我們?nèi)粘K龅降哪切╅T衛(wèi)一般,他們古怪偏執(zhí),死板守舊,我們可以利用反射基于類型擦除的認(rèn)識(shí),來(lái)繞過(guò)泛型中某些限制,現(xiàn)實(shí)生活中,也總會(huì)有調(diào)皮搗蛋者能夠基于對(duì)門衛(wèi)們生活作息的規(guī)律,選擇性地繞開(kāi)他們的監(jiān)視,另辟蹊徑溜進(jìn)或者溜出大門,然后揚(yáng)長(zhǎng)而去,剩下守衛(wèi)者一個(gè)孤獨(dú)的身影。 所以,我說(shuō)泛型,并不神秘,也不神奇。 |
|