轉(zhuǎn)載自這里,這哥們也是轉(zhuǎn)載但是沒注明....所以找不到原作者
1. 概述
泛型在java中有很重要的地位莱衩,在面向?qū)ο缶幊碳案鞣N設(shè)計(jì)模式中有非常廣泛的應(yīng)用爵嗅。
什么是泛型?為什么要使用泛型笨蚁?
泛型睹晒,即“參數(shù)化類型”。一提到參數(shù)括细,最熟悉的就是定義方法時(shí)有形參伪很,然后調(diào)用此方法時(shí)傳遞實(shí)參。那么參數(shù)化類型怎么理解呢奋单?
顧名思義锉试,就是將類型由原來的具體的類型參數(shù)化,類似于方法中的變量參數(shù)览濒,此時(shí)類型也定義成參數(shù)形式(可以稱之為類型形參)呆盖,
然后在使用/調(diào)用時(shí)傳入具體的類型(類型實(shí)參)。
泛型的本質(zhì)是為了參數(shù)化類型(在不創(chuàng)建新的類型的情況下贷笛,通過泛型指定的不同類型來控制形參具體限制的類型)应又。也就是說在泛型使用過程中,
操作的數(shù)據(jù)類型被指定為一個(gè)參數(shù)昨忆,這種參數(shù)類型可以用在類丁频、接口和方法中,分別被稱為泛型類邑贴、泛型接口席里、泛型方法。2. 一個(gè)栗子
一個(gè)被舉了無數(shù)次的例子:
List arrayList =new ArrayList();
arrayList.add("aaaa");
arrayList.add(100);for(inti = 0; i< arrayList.size();i++){
? ? String item = (String)arrayList.get(i);
? ? Log.d("泛型測試","item = " + item);
}
毫無疑問拢驾,程序的運(yùn)行結(jié)果會(huì)以崩潰結(jié)束:
java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.String
ArrayList可以存放任意類型奖磁,例子中添加了一個(gè)String類型,添加了一個(gè)Integer類型繁疤,再使用時(shí)都以String的方式使用咖为,因此程序崩潰了。為了解決類似這樣的問題(在編譯階段就可以解決)稠腊,泛型應(yīng)運(yùn)而生躁染。
我們將第一行聲明初始化list的代碼更改一下,編譯器會(huì)在編譯階段就能夠幫我們發(fā)現(xiàn)類似這樣的問題架忌。
List arrayList =newArrayList();
...//arrayList.add(100); 在編譯階段吞彤,編譯器就會(huì)報(bào)錯(cuò)
3. 特性
泛型只在編譯階段有效。看下面的代碼:
List stringArrayList =newArrayList();
List integerArrayList =newArrayList();
Class classStringArrayList = stringArrayList.getClass();
Class classIntegerArrayList = integerArrayList.getClass();if(classStringArrayList.equals(classIntegerArrayList)){
? ? Log.d("泛型測試","類型相同");
}
輸出結(jié)果:D/泛型測試: 類型相同饰恕。
通過上面的例子可以證明挠羔,在編譯之后程序會(huì)采取去泛型化的措施。也就是說Java中的泛型埋嵌,只在編譯階段有效破加。在編譯過程中,正確檢驗(yàn)泛型結(jié)果后雹嗦,會(huì)將泛型的相關(guān)信息擦出范舀,并且在對象進(jìn)入和離開方法的邊界處添加類型檢查和類型轉(zhuǎn)換的方法。也就是說俐银,泛型信息不會(huì)進(jìn)入到運(yùn)行時(shí)階段尿背。
對此總結(jié)成一句話:泛型類型在邏輯上看以看成是多個(gè)不同的類型端仰,實(shí)際上都是相同的基本類型捶惜。
泛型有三種使用方式,分別為:泛型類荔烧、泛型接口吱七、泛型方法
泛型類型用于類的定義中,被稱為泛型類鹤竭。通過泛型可以完成對一組類的操作對外開放相同的接口踊餐。最典型的就是各種容器類,如:List臀稚、Set吝岭、Map。
泛型類的最基本寫法(這么看可能會(huì)有點(diǎn)暈吧寺,會(huì)在下面的例子中詳解):
class類名稱 <泛型標(biāo)識(shí):可以隨便寫任意標(biāo)識(shí)號(hào)窜管,標(biāo)識(shí)指定的泛型的類型>{
? private泛型標(biāo)識(shí)/*(成員變量類型)*/ var;
? .....
? }
}
一個(gè)最普通的泛型類:
//此處T可以隨便寫為任意標(biāo)識(shí),常見的如T稚机、E幕帆、K、V等形式的參數(shù)常用于表示泛型//在實(shí)例化泛型類時(shí)赖条,必須指定T的具體類型publicclassGeneric{
? ? //key這個(gè)成員變量的類型為T,T的類型由外部指定? private T key;
? ? publicGeneric(T key) {//泛型構(gòu)造方法形參key的類型也為T失乾,T的類型由外部指定this.key = key;
? ? }
? ? publicT getKey(){//泛型方法getKey的返回值類型為T,T的類型由外部指定return key;
? ? }
}
//泛型的類型參數(shù)只能是類類型(包括自定義類)纬乍,不能是簡單類型//傳入的實(shí)參類型需與泛型的類型參數(shù)類型相同碱茁,即為Integer.Generic genericInteger =newGeneric(123456);//傳入的實(shí)參類型需與泛型的類型參數(shù)類型相同,即為String.Generic genericString =newGeneric("key_vlaue");
Log.d("泛型測試","key is " + genericInteger.getKey());
Log.d("泛型測試","key is " + genericString.getKey());
12-27 09:20:04.432 13063-13063/? D/泛型測試: key is 123456
12-27 09:20:04.432 13063-13063/? D/泛型測試: key is key_vlaue
定義的泛型類仿贬,就一定要傳入泛型類型實(shí)參么纽竣?并不是這樣,在使用泛型的時(shí)候如果傳入泛型實(shí)參诅蝶,則會(huì)根據(jù)傳入的泛型實(shí)參做相應(yīng)的限制退个,此時(shí)泛型才會(huì)起到本應(yīng)起到的限制作用募壕。如果不傳入泛型類型實(shí)參的話,在泛型類中使用泛型的方法或成員變量定義的類型可以為任何的類型语盈。
看一個(gè)例子:
Generic generic =newGeneric("111111");
Generic generic1 =newGeneric(4444);
Generic generic2 =newGeneric(55.55);
Generic generic3 =newGeneric(false);
Log.d("泛型測試","key is " + generic.getKey());
Log.d("泛型測試","key is " + generic1.getKey());
Log.d("泛型測試","key is " + generic2.getKey());
Log.d("泛型測試","key is " + generic3.getKey());
D/泛型測試: key is 111111D/泛型測試: key is 4444D/泛型測試: key is 55.55D/泛型測試: key isfalse
注意:
泛型的類型參數(shù)只能是類類型舱馅,不能是簡單類型。
不能對確切的泛型類型使用instanceof操作刀荒。如下面的操作是非法的代嗤,編譯時(shí)會(huì)出錯(cuò)。
if(ex_num?instanceof Generic<Number>){ }
4.4 泛型接口
泛型接口與泛型類的定義及使用基本相同缠借。泛型接口常被用在各種類的生產(chǎn)器中干毅,可以看一個(gè)例子:
//定義一個(gè)泛型接口publicinterfaceGenerator {
? ? public T next();
}
當(dāng)實(shí)現(xiàn)泛型接口的類,未傳入泛型實(shí)參時(shí):
/** * 未傳入泛型實(shí)參時(shí)泼返,與泛型類的定義相同硝逢,在聲明類的時(shí)候,需將泛型的聲明也一起加到類中
* 即:class FruitGenerator<T> implements Generator<T>{
* 如果不聲明泛型绅喉,如:class FruitGenerator implements Generator<T>渠鸽,編譯器會(huì)報(bào)錯(cuò):"Unknown class"
*/classFruitGeneratorimplementsGenerator{
? ? @Override
? ? public T next() {
? ? ? ? returnnull;
? ? }
}
當(dāng)實(shí)現(xiàn)泛型接口的類,傳入泛型實(shí)參時(shí):
/** * 傳入泛型實(shí)參時(shí):
* 定義一個(gè)生產(chǎn)器實(shí)現(xiàn)這個(gè)接口,雖然我們只創(chuàng)建了一個(gè)泛型接口Generator<T>
* 但是我們可以為T傳入無數(shù)個(gè)實(shí)參柴罐,形成無數(shù)種類型的Generator接口徽缚。
* 在實(shí)現(xiàn)類實(shí)現(xiàn)泛型接口時(shí),如已將泛型類型傳入實(shí)參類型革屠,則所有使用泛型的地方都要替換成傳入的實(shí)參類型
* 即:Generator<T>凿试,public T next();中的的T都要替換成傳入的String類型。
*/publicclassFruitGeneratorimplementsGenerator {
? ? privateString[] fruits =newString[]{"Apple", "Banana", "Pear"};
? ? @Override
? ? public String next() {
? ? ? ? Random rand =new Random();
? ? ? ? returnfruits[rand.nextInt(3)];
? ? }
}
4.5 泛型通配符
我們知道Ingeter是Number的一個(gè)子類似芝,同時(shí)在特性章節(jié)中我們也驗(yàn)證過Generic<Ingeter>與Generic<Number>實(shí)際上是相同的一種基本類型那婉。那么問題來了,在使用Generic<Number>作為形參的方法中国觉,能否使用Generic<Ingeter>的實(shí)例傳入呢吧恃?在邏輯上類似于Generic<Number>和Generic<Ingeter>是否可以看成具有父子關(guān)系的泛型類型呢?
為了弄清楚這個(gè)問題麻诀,我們使用Generic<T>這個(gè)泛型類繼續(xù)看下面的例子:
publicvoidshowKeyValue1(Generic obj){
? ? Log.d("泛型測試","key value is " + obj.getKey());
}
Generic gInteger =newGeneric(123);
Generic gNumber =newGeneric(456);
showKeyValue(gNumber);// showKeyValue這個(gè)方法編譯器會(huì)為我們報(bào)錯(cuò):Generic<java.lang.Integer> // cannot be applied to Generic<java.lang.Number>// showKeyValue(gInteger);
通過提示信息我們可以看到Generic<Integer>不能被看作為`Generic<Number>的子類痕寓。由此可以看出:同一種泛型可以對應(yīng)多個(gè)版本(因?yàn)閰?shù)類型是不確定的),不同版本的泛型類實(shí)例是不兼容的蝇闭。
回到上面的例子呻率,如何解決上面的問題?總不能為了定義一個(gè)新的方法來處理Generic<Integer>類型的類呻引,這顯然與java中的多臺(tái)理念相違背礼仗。因此我們需要一個(gè)在邏輯上可以表示同時(shí)是Generic<Integer>和Generic<Number>父類的引用類型。由此類型通配符應(yīng)運(yùn)而生。
我們可以將上面的方法改一下:
publicvoidshowKeyValue1(Generic obj){
? ? Log.d("泛型測試","key value is " + obj.getKey());
}
類型通配符一般是使用元践?代替具體的類型實(shí)參韭脊,注意了,此處’单旁?’是類型實(shí)參沪羔,而不是類型形參?。重要說三遍象浑!此處’蔫饰?’是類型實(shí)參,而不是類型形參?愉豺!?此處’篓吁?’是類型實(shí)參,而不是類型形參?蚪拦!再直白點(diǎn)的意思就是杖剪,此處的?和Number外盯、String摘盆、Integer一樣都是一種實(shí)際的類型,可以把饱苟?看成所有類型的父類。是一種真實(shí)的類型狼渊。
可以解決當(dāng)具體類型不確定的時(shí)候箱熬,這個(gè)通配符就是????;當(dāng)操作類型時(shí)狈邑,不需要使用類型的具體功能時(shí)城须,只使用Object類中的功能。那么可以用 ? 通配符來表未知類型米苹。
在java中,泛型類的定義非常簡單糕伐,但是泛型方法就比較復(fù)雜了。
尤其是我們見到的大多數(shù)泛型類中的成員方法也都使用了泛型蘸嘶,有的甚至泛型類中也包含著泛型方法良瞧,這樣在初學(xué)者中非常容易將泛型方法理解錯(cuò)了。
泛型類训唱,是在實(shí)例化類的時(shí)候指明泛型的具體類型褥蚯;泛型方法,是在調(diào)用方法的時(shí)候指明泛型的具體類型?况增。
/** * 泛型方法的基本介紹
* @param tClass 傳入的泛型實(shí)參
* @return T 返回值為T類型
* 說明:
*? ? 1)public 與 返回值中間<T>非常重要赞庶,可以理解為聲明此方法為泛型方法。
*? ? 2)只有聲明了<T>的方法才是泛型方法,泛型類中的使用了泛型的成員方法并不是泛型方法歧强。
*? ? 3)<T>表明該方法將使用泛型類型T澜薄,此時(shí)才可以在方法中使用泛型類型T。
*? ? 4)與泛型類的定義一樣摊册,此處T可以隨便寫為任意標(biāo)識(shí)表悬,常見的如T、E丧靡、K蟆沫、V等形式的參數(shù)常用于表示泛型。
*/public T genericMethod(Class tClass)throws InstantiationException ,
? IllegalAccessException{
? ? ? ? T instance = tClass.newInstance();
? ? ? ? return instance;
}
Object obj = genericMethod(Class.forName("com.test.test"));
4.6.1 泛型方法的基本用法
光看上面的例子有的同學(xué)可能依然會(huì)非常迷糊温治,我們再通過一個(gè)例子饭庞,把我泛型方法再總結(jié)一下。
publicclass GenericTest {
? //這個(gè)類是個(gè)泛型類熬荆,在上面已經(jīng)介紹過publicclassGeneric{? ?
? ? ? ? private T key;
? ? ? ? public Generic(T key) {
? ? ? ? ? ? this.key = key;
? ? ? ? }
? ? ? ? //我想說的其實(shí)是這個(gè)舟山,雖然在方法中使用了泛型,但是這并不是一個(gè)泛型方法卤恳。
? ? ? ? //這只是類中一個(gè)普通的成員方法累盗,只不過他的返回值是在聲明泛型類已經(jīng)聲明過的泛型。
? ? ? ? //所以在這個(gè)方法中才可以繼續(xù)使用 T 這個(gè)泛型突琳。public T getKey(){
? ? ? ? ? ? return key;
? ? ? ? }
? ? ? ? /**? ? ? ? * 這個(gè)方法顯然是有問題的若债,在編譯器會(huì)給我們提示這樣的錯(cuò)誤信息"cannot reslove symbol E"
? ? ? ? * 因?yàn)樵陬惖穆暶髦胁⑽绰暶鞣盒虴,所以在使用E做形參和返回值類型時(shí)拆融,編譯器會(huì)無法識(shí)別蠢琳。
? ? ? ? public E setKey(E key){
? ? ? ? ? ? this.key = keu
? ? ? ? }
? ? ? ? */? ? }
? ? /**
? ? * 這才是一個(gè)真正的泛型方法。
? ? * 首先在public與返回值之間的<T>必不可少镜豹,這表明這是一個(gè)泛型方法傲须,并且聲明了一個(gè)泛型T
? ? * 這個(gè)T可以出現(xiàn)在這個(gè)泛型方法的任意位置.
? ? * 泛型的數(shù)量也可以為任意多個(gè)
? ? *? ? 如:public <T,K> K showKeyName(Generic<T> container){
? ? *? ? ? ? ...
? ? *? ? ? ? }
? ? */public T showKeyName(Generic container){
? ? ? ? System.out.println("container key :" + container.getKey());
? ? ? ? //當(dāng)然這個(gè)例子舉的不太合適,只是為了說明泛型方法的特性趟脂。T test = container.getKey();
? ? ? ? return test;
? ? }
? ? //這也不是一個(gè)泛型方法泰讽,這就是一個(gè)普通的方法,只是使用了Generic<Number>這個(gè)泛型類做形參而已昔期。publicvoidshowKeyValue1(Generic obj){
? ? ? ? Log.d("泛型測試","key value is " + obj.getKey());
? ? }
? ? //這也不是一個(gè)泛型方法已卸,這也是一個(gè)普通的方法,只不過使用了泛型通配符?
? ? //同時(shí)這也印證了泛型通配符章節(jié)所描述的镇眷,?是一種類型實(shí)參咬最,可以看做為Number等所有類的父類publicvoidshowKeyValue2(Generic obj){
? ? ? ? Log.d("泛型測試","key value is " + obj.getKey());
? ? }
? ? /**? ? * 這個(gè)方法是有問題的,編譯器會(huì)為我們提示錯(cuò)誤信息:"UnKnown class 'E' "
? ? * 雖然我們聲明了<T>,也表明了這是一個(gè)可以處理泛型的類型的泛型方法欠动。
? ? * 但是只聲明了泛型類型T永乌,并未聲明泛型類型E惑申,因此編譯器并不知道該如何處理E這個(gè)類型。
? ? public <T> T showKeyName(Generic<E> container){
? ? ? ? ...
? ? }?
? ? *//**? ? * 這個(gè)方法也是有問題的翅雏,編譯器會(huì)為我們提示錯(cuò)誤信息:"UnKnown class 'T' "
? ? * 對于編譯器來說T這個(gè)類型并未項(xiàng)目中聲明過圈驼,因此編譯也不知道該如何編譯這個(gè)類。
? ? * 所以這也不是一個(gè)正確的泛型方法聲明望几。
? ? public void showkey(T genericObj){
? ? }
? ? */publicstaticvoid main(String[] args) {
? ? }
}
4.6.2 類中的泛型方法
當(dāng)然這并不是泛型方法的全部绩脆,泛型方法可以出現(xiàn)雜任何地方和任何場景中使用。但是有一種情況是非常特殊的橄抹,當(dāng)泛型方法出現(xiàn)在泛型類中時(shí)靴迫,我們再通過一個(gè)例子看一下
publicclass GenericFruit {
? ? class Fruit{
? ? ? ? @Override
? ? ? ? public String toString() {
? ? ? ? ? ? return"fruit";
? ? ? ? }
? ? }
? ? classAppleextends Fruit{
? ? ? ? @Override
? ? ? ? public String toString() {
? ? ? ? ? ? return"apple";
? ? ? ? }
? ? }
? ? class Person{
? ? ? ? @Override
? ? ? ? public String toString() {
? ? ? ? ? ? return"Person";
? ? ? ? }
? ? }
? ? classGenerateTest{
? ? ? ? publicvoid show_1(T t){
? ? ? ? ? ? System.out.println(t.toString());
? ? ? ? }
? ? ? ? //在泛型類中聲明了一個(gè)泛型方法,使用泛型E楼誓,這種泛型E可以為任意類型玉锌。可以類型與T相同疟羹,也可以不同主守。
? ? ? ? //由于泛型方法在聲明的時(shí)候會(huì)聲明泛型<E>,因此即使在泛型類中并未聲明泛型榄融,編譯器也能夠正確識(shí)別泛型方法中識(shí)別的泛型参淫。publicvoid show_3(E t){
? ? ? ? ? ? System.out.println(t.toString());
? ? ? ? }
? ? ? ? //在泛型類中聲明了一個(gè)泛型方法,使用泛型T愧杯,注意這個(gè)T是一種全新的類型涎才,可以與泛型類中聲明的T不是同一種類型。publicvoid show_2(T t){
? ? ? ? ? ? System.out.println(t.toString());
? ? ? ? }
? ? }
? ? publicstaticvoid main(String[] args) {
? ? ? ? Apple apple =new Apple();
? ? ? ? Person person =new Person();
? ? ? ? GenerateTest generateTest =newGenerateTest();
? ? ? ? //apple是Fruit的子類民效,所以這里可以? ? ? ? generateTest.show_1(apple);
? ? ? ? //編譯器會(huì)報(bào)錯(cuò)憔维,因?yàn)榉盒皖愋蛯?shí)參指定的是Fruit,而傳入的實(shí)參類是Person
? ? ? ? //generateTest.show_1(person);
? ? ? ? //使用這兩個(gè)方法都可以成功? ? ? ? generateTest.show_2(apple);
? ? ? ? generateTest.show_2(person);
? ? ? ? //使用這兩個(gè)方法也都可以成功? ? ? ? generateTest.show_3(apple);
? ? ? ? generateTest.show_3(person);
? ? }
}
4.6.3 泛型方法與可變參數(shù)
再看一個(gè)泛型方法和可變參數(shù)的例子:
publicvoid printMsg( T... args){
? ? for(T t : args){
? ? ? ? Log.d("泛型測試","t is " + t);
? ? }
}
printMsg("111",222,"aaaa","2323.4",55.55);
4.6.4 靜態(tài)方法與泛型
靜態(tài)方法有一種情況需要注意一下畏邢,那就是在類中的靜態(tài)方法使用泛型:靜態(tài)方法無法訪問類上定義的泛型;如果靜態(tài)方法操作的引用數(shù)據(jù)類型不確定的時(shí)候检吆,必須要將泛型定義在方法上舒萎。
即:如果靜態(tài)方法要使用泛型的話,必須將靜態(tài)方法也定義成泛型方法?蹭沛。
publicclassStaticGenerator {
? ? ....
? ? ....
? ? /**? ? * 如果在類中定義使用泛型的靜態(tài)方法臂寝,需要添加額外的泛型聲明(將這個(gè)方法定義成泛型方法)
? ? * 即使靜態(tài)方法要使用泛型類中已經(jīng)聲明過的泛型也不可以。
? ? * 如:public static void show(T t){..},此時(shí)編譯器會(huì)提示錯(cuò)誤信息:
? ? ? ? ? "StaticGenerator cannot be refrenced from static context"
? ? */publicstaticvoid show(T t){
? ? }
}
4.6.5 泛型方法總結(jié)
泛型方法能使方法獨(dú)立于類而產(chǎn)生變化摊灭,以下是一個(gè)基本的指導(dǎo)原則:
無論何時(shí)咆贬,如果你能做到,你就該盡量使用泛型方法帚呼。也就是說掏缎,如果使用泛型方法將整個(gè)類泛型化皱蹦,
那么就應(yīng)該使用泛型方法。另外對于一個(gè)static的方法而已眷蜈,無法訪問泛型類型的參數(shù)沪哺。
所以如果static方法要使用泛型能力,就必須使其成為泛型方法酌儒。
4.6 泛型上下邊界
在使用泛型的時(shí)候辜妓,我們還可以為傳入的泛型類型實(shí)參進(jìn)行上下邊界的限制,如:類型實(shí)參只準(zhǔn)傳入某種類型的父類或某種類型的子類忌怎。
為泛型添加上邊界籍滴,即傳入的類型實(shí)參必須是指定類型的子類型
publicvoidshowKeyValue1(Generic obj){
? ? Log.d("泛型測試","key value is " + obj.getKey());
}
Generic generic1 =newGeneric("11111");
Generic generic2 =newGeneric(2222);
Generic generic3 =newGeneric(2.4f);
Generic generic4 =newGeneric(2.56);//這一行代碼編譯器會(huì)提示錯(cuò)誤,因?yàn)镾tring類型并不是Number類型的子類//showKeyValue1(generic1);showKeyValue1(generic2);
showKeyValue1(generic3);
showKeyValue1(generic4);
如果我們把泛型類的定義也改一下:
publicclassGeneric{
? ? private T key;
? ? public Generic(T key) {
? ? ? ? this.key = key;
? ? }
? ? public T getKey(){
? ? ? ? return key;
? ? }
}
//這一行代碼也會(huì)報(bào)錯(cuò)榴啸,因?yàn)镾tring不是Number的子類Generic generic1 =newGeneric("11111");
再來一個(gè)泛型方法的例子:
//在泛型方法中添加上下邊界限制的時(shí)候孽惰,必須在權(quán)限聲明與返回值之間的<T>上添加上下邊界,即在泛型聲明的時(shí)候添加//public <T> T showKeyName(Generic<T extends Number> container)插掂,編譯器會(huì)報(bào)錯(cuò):"Unexpected bound"public T showKeyName(Generic container){
? ? System.out.println("container key :" + container.getKey());
? ? T test = container.getKey();
? ? return test;
}
通過上面的兩個(gè)例子可以看出:泛型的上下邊界添加灰瞻,必須與泛型的聲明在一起?。
看到了很多文章中都會(huì)提起泛型數(shù)組辅甥,經(jīng)過查看sun的說明文檔酝润,在java中是”不能創(chuàng)建一個(gè)確切的泛型類型的數(shù)組”的。
也就是說下面的這個(gè)例子是不可以的:
List[] ls =newArrayList[10];
而使用通配符創(chuàng)建泛型數(shù)組是可以的璃弄,如下面這個(gè)例子:
List[] ls =newArrayList[10];
這樣也是可以的:
List[] ls =newArrayList[10];
下面使用Sun的一篇文檔的一個(gè)例子來說明這個(gè)問題:
List[] lsa =newList[10];// Not really allowed.? ? Object o = lsa;? ?
Object[] oa = (Object[]) o;? ?
List li =newArrayList();? ?
li.add(newInteger(3));? ?
oa[1] = li;// Unsound, but passes run time store check? ? String s = lsa[1].get(0);// Run-time error: ClassCastException.
這種情況下要销,由于JVM泛型的擦除機(jī)制,在運(yùn)行時(shí)JVM是不知道泛型信息的夏块,所以可以給oa[1]賦上一個(gè)ArrayList而不會(huì)出現(xiàn)異常疏咐,
但是在取出數(shù)據(jù)的時(shí)候卻要做一次類型轉(zhuǎn)換,所以就會(huì)出現(xiàn)ClassCastException脐供,如果可以進(jìn)行泛型數(shù)組的聲明浑塞,
上面說的這種情況在編譯期將不會(huì)出現(xiàn)任何的警告和錯(cuò)誤,只有在運(yùn)行時(shí)才會(huì)出錯(cuò)政己。而對泛型數(shù)組的聲明進(jìn)行限制酌壕,對于這樣的情況,可以在編譯期提示代碼有類型安全問題歇由,比沒有任何提示要強(qiáng)很多卵牍。
下面采用通配符的方式是被允許的:數(shù)組的類型不可以是類型變量,除非是采用通配符的方式沦泌,因?yàn)閷τ谕ㄅ浞姆绞胶迹詈笕〕鰯?shù)據(jù)是要做顯式的類型轉(zhuǎn)換的。
List[] lsa =newList[10];// OK, array of unbounded wildcard type.? ? Object o = lsa;? ?
Object[] oa = (Object[]) o;? ?
List li =newArrayList();? ?
li.add(newInteger(3));? ?
oa[1] = li;// Correct.? ? Integer i = (Integer) lsa[1].get(0);// OK
5. 最后
本文中的例子主要是為了闡述泛型中的一些思想而簡單舉出的谢谦,并不一定有著實(shí)際的可用性释牺。另外萝衩,一提到泛型,相信大家用到最多的就是在集合中船侧,其實(shí)欠气,在實(shí)際的編程過程中,自己可以使用泛型去簡化開發(fā)镜撩,且能很好的保證代碼質(zhì)量预柒。