1摄狱、何為泛型
首先泛型的本質(zhì)便是類型參數(shù)化,通俗的說就是用一個變量來表示類型无午,這個類型可以是String,Integer等等不確定媒役,表明可接受的類型,原理類似如下代碼
int pattern; //聲明一個變量未賦值,pattern可以看作是泛型
pattern = 4;
pattern = 5;//4和5就可以看作是String和Integer
泛型的具體形式見泛型類宪迟、泛型方法
*泛型方法舉例代碼如下
public <T> void show()
{
operation about T...
}
泛型參數(shù)類型聲明必須在返回類型之前
2酣衷、為何要引入泛型,即泛型與Object的優(yōu)勢
由于泛型可以接受多個參數(shù)次泽,而Object經(jīng)過強制類型轉(zhuǎn)換可以轉(zhuǎn)換為任何類型穿仪,既然二者都具有相同的作用,為何還要引進(jìn)泛型呢意荤?
解答:泛型可以把使用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);//在編譯時沒問題钠龙,但在運行時出現(xiàn)問題
首先聲明無泛型的ArrayList時,其默認(rèn)的原始類型是Object數(shù)組御铃,既然為Object類型碴里,就可以接受任意數(shù)據(jù)的賦值,因此編譯時沒有問題上真,但是在運行時,Integer強轉(zhuǎn)成String,肯定會出現(xiàn)ClassCastException.因此泛型的引入增強了安全性咬腋,把類轉(zhuǎn)換異常提前到了編譯時期。
3睡互、類型擦除和原始類型
類型擦除的由來
在JAVA的虛擬機中并不存在泛型根竿,泛型只是為了完善java體系,增加程序員編程的便捷性以及安全性而創(chuàng)建的一種機制就珠,在JAVA虛擬機中對應(yīng)泛型的都是確定的類型寇壳,在編寫泛型代碼后,java虛擬中會把這些泛型參數(shù)類型都擦除妻怎,用相應(yīng)的確定類型來代替壳炎,代替的這一動作叫做類型擦除,而用于替代的類型稱為原始類型逼侦,在類型擦除過程中匿辩,一般使用第一個限定的類型來替換腰耙,若無限定則使用Object.
泛型方法的翻譯
因此在虛擬機翻譯泛型方法中,引入了橋方法稼病,及在類型擦除后的show(Object t)中調(diào)用另一個方法选侨,代碼如下:
public void show(Object t){
show((String) t);
}
4、泛型限定
泛型限定是通過然走?(通配符)來實現(xiàn)的侵俗,表示可以接受任意類型,那一定有人有疑問丰刊,那隘谣?和T(二者單獨使用時)有啥區(qū)別了,其實區(qū)別也不是很大啄巧,僅僅在對參數(shù)類型的使用上寻歧。例如:
? extends SomeClass 這種限定,說明的是只能接收SomeClass及其子類類型秩仆,所謂的“上限”
? super SomeClass 這種限定码泛,說明只能接收SomeClass及其父類類型,所謂的“下限”
一下舉例? extends SomeClass說明一下這類限定的一種應(yīng)用方式
由于泛型參數(shù)類型可以表示任意類型的類類型澄耍,若T要引用compareTo方法噪珊,如何保證在T類中定義了compareTo方法呢?利用如下代碼:
public <T extends Comparable> shwo(T a, T b){
int num = a.compareTo(b);
}
此處用于限定T類型繼承自Comparable,因為T類型可以調(diào)用compareTo方法齐莲。
可以有多個類型限定痢站,例如:
<T extends Comparable & Serializable>
關(guān)于泛型類型限定的“繼承”誤區(qū)
總有些人誤把類型的限定當(dāng)作繼承,比如:
//類型是這樣的
<Student extends Person>
//然后出現(xiàn)此類錯誤
ArrayList<Person> al = new ArrayList<Student>();
此處的<Person>, <Student>作為泛型的意思是ArrayList容器的接收類型殿衰,用一個簡單的例子來說明
ArrayList是一個大型養(yǎng)殖場朱庆,<Person>表明的是他能夠接收動物,而上邊的new語句生成的是一個只能夠接收豬的養(yǎng)殖場(ArrayList<Student>),即把一個大型養(yǎng)殖場建造成了一個養(yǎng)豬場闷祥,若是繼承的大型養(yǎng)殖場肯定是還能接受狗娱颊、羊....的,但是現(xiàn)在建造成了養(yǎng)豬場,那還能接受別的動物么维蒙?所以這肯定是錯誤的用法掰吕!簡而言之果覆,泛型new時兩邊的類型參數(shù)必須一致颅痊。
5、泛型的一些基本規(guī)則約束
泛型的類型參數(shù)必須為類的引用局待,不能用基本類型(int, short, long, byte, float, double, char, boolean)
泛型是類型的參數(shù)化斑响,在使用時可以用作不同類型(此處在說泛型類時會詳細(xì)說明)
泛型的類型參數(shù)可以有多個,代碼舉例如下:
public <T, E> void show(){ coding operation..... }
泛型可以使用extends, super, ?(通配符)來對類型參數(shù)進(jìn)行限定
由于類型擦除钳榨,運行時類型查詢只適用于原始類型舰罚,比如instanceof、getClass()薛耻、強制類型轉(zhuǎn)換营罢,a instanceof (Pair<Employe>),在類型擦除后便是 a instanceof Pair,因此以上運行的一些操作在虛擬機中操作都是對原始類型進(jìn)行操作,無論寫的多么虛幻饼齿,都逃不出類型擦除饲漾,因為在虛擬機種并不存在泛型。
不能創(chuàng)建參數(shù)化類型的數(shù)組
不能實例化類型變量僚楞,及不能出現(xiàn)以下的類似代碼
T t = new T();//或T.Class
因為在類型擦除后,便是Object t = new Object();與用意不符合枉层,即本意肯定不是要調(diào)用Object.
不能再靜態(tài)域或方法中出現(xiàn)參數(shù)類型
但是這樣的代碼就是正確的
class Test<T>{public static <T> T show() { action }}
因為此處的靜態(tài)方法是泛型方法,可以使用.
不能拋出或捕獲泛型類的實例
+不能拋出不能捕獲泛型類對象
+泛型類不能擴展Throwable癌淮,注意是類不能繼承Throwable躺坟,類型參數(shù)的限定還是可以的。
+catch子句不能使用類型變量乳蓄,如下代碼:
此處的錯誤的原因不能存在同一個方法,在類型擦除后,Pair的方法為美侦,public boolean equals(Object value),這與從Object.class中繼承下來的equals(Object obj)沖突产舞。
一個類不能成為兩個接口類型的子類,而這兩個接口是同一接口的不同參數(shù)化菠剩。
例如:
class Calendar implements coparable<Calendar>{}class GregorianCalendar extends Calendar implements Comparable<GregorianCalendar>{} //error
當(dāng)類型擦除后易猫,Calendar實現(xiàn)的是Comparable,而GregorianCalendar繼承了Calendar,又去實現(xiàn)Comparable,必然出錯具壮!
———————————————————————————————————————————————————————————————————————————————
先總結(jié)到此處准颓。