Java泛型深入理解小總結

1、何為泛型
首先泛型的本質便是類型參數化,通俗的說就是用一個變量來表示類型,這個類型可以是String,Integer等等不確定,表明可接受的類型,原理類似如下代碼
int pattern; //聲明一個變量未賦值,pattern可以看作是泛型
pattern = 4;
pattern = 5;//4和5就可以看作是String和Integer

泛型的具體形式見泛型類、泛型方法

泛型類形式如下

*泛型方法舉例代碼如下
public <T> void show()
{
operation about T...
}
泛型參數類型聲明必須在返回類型之前
2、為何要引入泛型,即泛型與Object的優勢
由于泛型可以接受多個參數,而Object經過強制類型轉換可以轉換為任何類型,既然二者都具有相同的作用,為何還要引進泛型呢?
解答:泛型可以把使用Object的錯誤提前到編譯后,而不是運行后,提升安全性。以下用帶泛型的ArrayList和不帶泛型的Arraylist舉例說明
代碼1:
ArrayList al = new ArrayList();al.add("hello");
al.add(4);//自動裝箱
String s1 = (String)al.get(0);
String s2 = (String)al.get(1);//在編譯時沒問題,但在運行時出現問題

首先聲明無泛型的ArrayList時,其默認的原始類型是Object數組,既然為Object類型,就可以接受任意數據的賦值,因此編譯時沒有問題,但是在運行時,Integer強轉成String,肯定會出現ClassCastException.因此泛型的引入增強了安全性,把類轉換異常提前到了編譯時期。
3、類型擦除和原始類型
  類型擦除的由來
在JAVA的虛擬機中并不存在泛型,泛型只是為了完善java體系,增加程序員編程的便捷性以及安全性而創建的一種機制,在JAVA虛擬機中對應泛型的都是確定的類型,在編寫泛型代碼后,java虛擬中會把這些泛型參數類型都擦除,用相應的確定類型來代替,代替的這一動作叫做類型擦除,而用于替代的類型稱為原始類型,在類型擦除過程中,一般使用第一個限定的類型來替換,若無限定則使用Object.

對泛型類的翻譯,泛型類(不帶泛型限定)代碼:

虛擬機進行翻譯后的原始類型:
泛型類(帶泛型限定)代碼: 
虛擬機進行翻譯后的原始類型:

泛型方法的翻譯


由于TestDemo繼承Test<String>,但是Test在類型擦除后還有一個public void Show(Object t),這和那個show(String t)出現重載,但是本意卻是沒有show(Object t)的,
因此在虛擬機翻譯泛型方法中,引入了橋方法,及在類型擦除后的show(Object t)中調用另一個方法,代碼如下:
public void show(Object t){
show((String) t);
}
4、泛型限定
  泛型限定是通過?(通配符)來實現的,表示可以接受任意類型,那一定有人有疑問,那?和T(二者單獨使用時)有啥區別了,其實區別也不是很大,僅僅在對參數類型的使用上。例如:

? extends SomeClass 這種限定,說明的是只能接收SomeClass及其子類類型,所謂的“上限”
  ? super SomeClass 這種限定,說明只能接收SomeClass及其父類類型,所謂的“下限”
一下舉例? extends SomeClass說明一下這類限定的一種應用方式
由于泛型參數類型可以表示任意類型的類類型,若T要引用compareTo方法,如何保證在T類中定義了compareTo方法呢?利用如下代碼:
public <T extends Comparable> shwo(T a, T b){
int num = a.compareTo(b);
}
此處用于限定T類型繼承自Comparable,因為T類型可以調用compareTo方法。
  可以有多個類型限定,例如:
<T extends Comparable & Serializable>

這種書寫方式為何把comparable寫在前邊?因為由于類型擦除的問題,原始類型是由Comparable替換的,因此寫在前邊的是類中存在此類型的泛型方法放在前邊,避免調用方法時類型的強制轉換,提高效率。

關于泛型類型限定的“繼承”誤區
總有些人誤把類型的限定當作繼承,比如:
//類型是這樣的
<Student extends Person>
//然后出現此類錯誤
ArrayList<Person> al = new ArrayList<Student>();
此處的<Person>, <Student>作為泛型的意思是ArrayList容器的接收類型,用一個簡單的例子來說明
ArrayList是一個大型養殖場,<Person>表明的是他能夠接收動物,而上邊的new語句生成的是一個只能夠接收豬的養殖場(ArrayList<Student>),即把一個大型養殖場建造成了一個養豬場,若是繼承的大型養殖場肯定是還能接受狗、羊....的,但是現在建造成了養豬場,那還能接受別的動物么?所以這肯定是錯誤的用法!簡而言之,泛型new時兩邊的類型參數必須一致。
5、泛型的一些基本規則約束
  泛型的類型參數必須為類的引用,不能用基本類型(int, short, long, byte, float, double, char, boolean)
  泛型是類型的參數化,在使用時可以用作不同類型(此處在說泛型類時會詳細說明)
  泛型的類型參數可以有多個,代碼舉例如下:
public <T, E> void show(){ coding operation..... }

泛型可以使用extends, super, ?(通配符)來對類型參數進行限定
  由于類型擦除,運行時類型查詢只適用于原始類型,比如instanceof、getClass()、強制類型轉換,a instanceof (Pair<Employe>),在類型擦除后便是 a instanceof Pair,因此以上運行的一些操作在虛擬機中操作都是對原始類型進行操作,無論寫的多么虛幻,都逃不出類型擦除,因為在虛擬機種并不存在泛型。
  不能創建參數化類型的數組

例如寫如下代碼:
由于Object可以接收任何類型,在里邊存入 new Pari<Employe>時,沒有任何問題,但是當取出的時候會出現ClassCastException,因此不能創建參數化類型數組。
  不能實例化類型變量,及不能出現以下的類似代碼
T t = new T();//或T.Class

因為在類型擦除后,便是Object t = new Object();與用意不符合,即本意肯定不是要調用Object.
  不能再靜態域或方法中出現參數類型

例如如下錯誤代碼:
首先方法是一個返回類型為T的普通方法,而非泛型方法,這和在靜態方法中使用非靜態參數是一樣的,靜態方法是先于對象而存在內存中的,因此在編譯的時候,T的類型無法確定,一定會出現“Cannot make a static reference to a non_static reference”這樣類似的錯誤。
但是這樣的代碼就是正確的
class Test<T>{public static <T> T show() { action }}
因為此處的靜態方法是泛型方法,可以使用.
  不能拋出或捕獲泛型類的實例
    +不能拋出不能捕獲泛型類對象
    +泛型類不能擴展Throwable,注意是類不能繼承Throwable,類型參數的限定還是可以的。
    +catch子句不能使用類型變量,如下代碼:

類型擦除后的沖突注意,例如:

此處的錯誤的原因不能存在同一個方法,在類型擦除后,Pair的方法為,public boolean equals(Object value),這與從Object.class中繼承下來的equals(Object obj)沖突。
  一個類不能成為兩個接口類型的子類,而這兩個接口是同一接口的不同參數化。
例如:
class Calendar implements coparable<Calendar>{}class GregorianCalendar extends Calendar implements Comparable<GregorianCalendar>{} //error

當類型擦除后,Calendar實現的是Comparable,而GregorianCalendar繼承了Calendar,又去實現Comparable,必然出錯!
———————————————————————————————————————————————————————————————————————————————
先總結到此處。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,983評論 6 537
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,772評論 3 422
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,947評論 0 381
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,201評論 1 315
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,960評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,350評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,406評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,549評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,104評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,914評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,089評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,647評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,340評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,753評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,007評論 1 289
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,834評論 3 395
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,106評論 2 375

推薦閱讀更多精彩內容

  • 泛型是Java 1.5引入的新特性。泛型的本質是參數化類型,這種參數類型可以用在類、變量、接口和方法的創建中,分別...
    何時不晚閱讀 3,047評論 0 2
  • 引言:泛型一直是困擾自己的一個難題,但是泛型有時一個面試時老生常談的問題;今天作者就通過查閱相關資料簡單談談自己對...
    cp_insist閱讀 1,857評論 0 4
  • 在之前的文章中分析過了多態,可以知道多態本身是一種泛化機制,它通過基類或者接口來設計,使程序擁有一定的靈活性,但是...
    _小二_閱讀 694評論 0 0
  • [TOC] 深入理解 Java 泛型 概述 泛型的本質是參數化類型,通常用于輸入參數、存儲類型不確定的場景。相比于...
    albon閱讀 5,333評論 0 7
  • Why ——引入泛型機制的原因 假如我們想要實現一個String數組,并且要求它可以動態改變大小,這時我們都會想到...
    absfree閱讀 5,134評論 1 6