初探Java中的泛型
泛型是一個很有意思也很重要的概念,本篇將簡單介紹Java中的泛型特性,主要從以下角度講解:
1.什么是泛型。
2.如何使用泛型。
3.泛型的好處。
1.什么是泛型?
泛型,字面意思便是參數(shù)化類型,平時所面對的類型一般都是具體的類型,如果String,Integer,Double,而泛型則是把所操作的數(shù)據(jù)類型當(dāng)作一個參數(shù)。如,ArrayList<String>(),通過傳入不同的類型來指定容器中存儲的類型,而不用為不同的類型創(chuàng)建不同的類,這種參數(shù)類型可以用在類、接口和方法的創(chuàng)建中,分別稱為泛型類、泛型接口、泛型方法。
2.如何使用泛型?
我們先來看看泛型是什么樣子的:
public interface List<E> { void add(E); Iterator<E> iterator();}
這是List接口,這里用E來代替具體類型,這樣就可以往里面?zhèn)魅肴我忸愋停苍S你要問了,直接使用Object不好嗎?我們來用一個栗子比較一下:
先用非泛型方式來實現(xiàn)一下:
public class ObjHolder { private Object a; public ObjHolder(Object a) { this.a = a; } public void set(Object a) { this.a = a; } public Object get(){ return a; } public static void main(String[] args) { ObjHolder holderA = new ObjHolder('Frank'); System.out.println((String) holderA.get()); holderA.set(233); System.out.println((Integer) holderA.get()); }}
這樣就實現(xiàn)了一個包裝類,可以用來存取一個任意類型的對象。但是每次取出來都需要進(jìn)行類型轉(zhuǎn)化,如果方法的參數(shù)類型是ObjHolder的話,無法知道它里面存放的對象的確切類型,這樣就反而帶來很多不必要的麻煩。
現(xiàn)在來看一下用泛型實現(xiàn)是怎樣的:
public class GenericHolder<T> { private T obj; public GenericHolder(T obj){ this.obj = obj; } public T getObj() { return obj; } public void setObj(T obj) { this.obj = obj; } public static void main(String[] args) { GenericHolder<String> holderA = new GenericHolder<String>('Frank'); System.out.println(holderA.getObj());//holderA.set(233);無法編譯通過,因為只能往holderA中存入String類型 GenericHolder<Integer> holderB = new GenericHolder<Integer>(233); System.out.println(holderB.getObj()); }}
這樣通過傳入類型信息如String和Integer,來代替其中的泛型參數(shù)T,這里的T可以理解為一個占位符,用其他字母也是可以的,一旦傳入具體類型,如String,則所有使用T的地方都會用String類型替換。
對比一下上面兩種方式,區(qū)別在哪呢?打個比方,不用泛型的實現(xiàn)方式,相當(dāng)于一個袋子,里面可以裝任意類型的黑盒子,你什么都可以往里放,但是你可能不知道你下一個取出來的是什么東西,而泛型的實現(xiàn)方式,相當(dāng)于一個貼了標(biāo)簽的黑盒子,標(biāo)簽上可以寫任何信息,如寫上水果,那么這個盒子就只能裝水果,你也會知道每次取出來的肯定是水果而不是其它東西,同理類似如寫上雜糧,那么這個袋子就只能用來裝雜糧,但其實上都是同一種袋子,并不是為每一種類型的東西準(zhǔn)備一種袋子。(因為Java的泛型使用了類型擦除機(jī)制,至于類型擦除是什么,暫時不做過多介紹,以后會有文章做更詳細(xì)的說明)。
泛型被廣泛應(yīng)用在容器類中,如ArrayList<T>() 表示用于存儲特定類型的數(shù)組,除此之外,還有很多泛型接口,如Comparable<T>。使用泛型能帶來極大的便利性。
在泛型中可以對類型進(jìn)行限制,如:<T extends Comparable<T>>表示只能傳遞已經(jīng)實現(xiàn)了Comparable接口的類型對象,這里是使用extends而不是implement,而且對于接口也只能寫一個。<T extends Number>表示只能接收Number類或者其子類的對象。與之相反的邊界通配符是super,如:<T extends Phone>表示只能接收類型為Phone或其父類的對象。
在使用extends和super的時候需要特別注意,因為使用它們是有副作用的,比如:
List<T extends Number> list = new ArrayList<Number>();list.add(4.0);//編譯錯誤list.add(3);//編譯錯誤
因為泛型是為了類型安全設(shè)計的,如果往List<? extends Number> list 塞值的話,在取的時候就無法確認(rèn)它到底是什么類型了,編譯器只知道它是Number類型或者它的派生類型,但無法確定是哪個具體類型。通配符T表示其中存的都是同一種類型,因此使用extend下邊界的話是無法進(jìn)行存操作的。同理super下邊界是不能取值的。
那什么時候該用extends,什么時候該用super呢?先說結(jié)論:
PECS原則:
頻繁往外讀取內(nèi)容的,適合用上界Extends。 經(jīng)常往里插入的,適合用下界Super。3.泛型的好處?
泛型看起來很炫酷,但初看起來,好像沒什么卵用?客官且慢,進(jìn)屋里坐(滑稽)。
使用泛型的好處我們來一項一項列出來:
1,類型安全。
這是最顯而易見的,泛型的主要目標(biāo)是提高 Java 程序的類型安全。通過使用泛型定義的變量的類型限制,可以很容易實現(xiàn)編譯期間的類型檢測,避免了大量因為使用Object帶來的不必要的類型錯誤。
沒有泛型,這些對Object變量的類型假設(shè)就只存在于程序員的頭腦中(或者如果幸運的話,還存在于代碼注釋中),而且每次使用前還需要進(jìn)行不安全的強(qiáng)制類型轉(zhuǎn)換。
2,代碼復(fù)用。
泛型的一個很大好處就是增加了代碼的復(fù)用性,比如上面的 GenericHolder 類,就能存取任意類型的對象,而不用為每種類型寫一個包裝類。
3,潛在的性能收益。
泛型為較大的優(yōu)化帶來可能。在泛型的初始實現(xiàn)中,編譯器將強(qiáng)制類型轉(zhuǎn)換(沒有泛型的話,程序員會指定這些強(qiáng)制類型轉(zhuǎn)換)插入生成的字節(jié)碼中。但是更多類型信息可用于編譯器這一事實,為未來版本的 JVM 的優(yōu)化帶來可能。由于泛型的實現(xiàn)方式,支持泛型(幾乎)不需要 JVM 或類文件更改。所有工作都在編譯器中完成,編譯器生成類似于沒有泛型(和強(qiáng)制類型轉(zhuǎn)換)時所寫的代碼,只是更能確保類型安全而已。Java語言引入泛型的好處是安全簡單。泛型的好處是在編譯的時候檢查類型安全,并且所有的強(qiáng)制轉(zhuǎn)換都是自動和隱式的,提高代碼的重用率。
至此,本篇講解完畢,如果想要更好的理解,還需要多寫代碼,在實踐中去應(yīng)用。
歡迎大家繼續(xù)關(guān)注!
以上就是初探Java中的泛型的詳細(xì)內(nèi)容,更多關(guān)于Java 泛型的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. .NET中l(wèi)ambda表達(dá)式合并問題及解決方法2. JSP數(shù)據(jù)交互實現(xiàn)過程解析3. 淺談python出錯時traceback的解讀4. 利用promise及參數(shù)解構(gòu)封裝ajax請求的方法5. Python importlib動態(tài)導(dǎo)入模塊實現(xiàn)代碼6. python matplotlib:plt.scatter() 大小和顏色參數(shù)詳解7. windows服務(wù)器使用IIS時thinkphp搜索中文無效問題8. ASP 信息提示函數(shù)并作返回或者轉(zhuǎn)向9. 在Android中使用WebSocket實現(xiàn)消息通信的方法詳解10. Nginx+php配置文件及原理解析
