中文字幕精品亚洲无线码二区,国产黄a三级三级三级看三级,亚洲七七久久桃花影院,丰满少妇被猛烈进入,国产小视频在线观看网站

java 泛型詳(xiang)解(jie)-絕(jue)對是對泛型方(fang)法(fa)講(jiang)解(jie)最詳(xiang)細的(de),沒(mei)有(you)之一

對(dui)java的(de)泛型(xing)特性(xing)的(de)了解僅限于表面的(de)淺(qian)淺(qian)一(yi)層,直到在學習設計模式時(shi)發現有(you)不了解的(de)用法(fa),才想起詳細的(de)記(ji)錄一(yi)下。

本文參考java 泛型(xing)詳解、Java中的(de)泛型(xing)方(fang)法(fa)、 java泛型(xing)詳解

1. 概述

泛型(xing)在java中有(you)很重要的(de)地位,在面(mian)向對象(xiang)編程及各種(zhong)設計(ji)模式中有(you)非常廣泛的(de)應(ying)用(yong)。

什(shen)么(me)是(shi)泛型?為什(shen)么(me)要使用泛型?

泛(fan)型,即“參數化類(lei)型”。一提到參數,最(zui)熟悉(xi)的就(jiu)是(shi)定(ding)義方(fang)法(fa)時有形參,然后調用此方(fang)法(fa)時傳遞實參。那(nei)么參數化類(lei)型怎么理(li)解呢(ni)?

顧名思義,就是將(jiang)類(lei)(lei)型由(you)原來的具體的類(lei)(lei)型參(can)數化,類(lei)(lei)似于方法中的變量參(can)數,此時類(lei)(lei)型也(ye)定(ding)義成參(can)數形式(可以(yi)稱之(zhi)為類(lei)(lei)型形參(can)),

然后在使用(yong)/調(diao)用(yong)時(shi)傳入(ru)具體的(de)(de)(de)類(lei)(lei)型(xing)(xing)(xing)(類(lei)(lei)型(xing)(xing)(xing)實(shi)參)。 泛(fan)(fan)(fan)型(xing)(xing)(xing)的(de)(de)(de)本質(zhi)是(shi)為了(le)參數化類(lei)(lei)型(xing)(xing)(xing)(在不(bu)創建新(xin)的(de)(de)(de)類(lei)(lei)型(xing)(xing)(xing)的(de)(de)(de)情況下,通(tong)過泛(fan)(fan)(fan)型(xing)(xing)(xing)指定(ding)的(de)(de)(de)不(bu)同類(lei)(lei)型(xing)(xing)(xing)來控制(zhi)形參具體限制(zhi)的(de)(de)(de)類(lei)(lei)型(xing)(xing)(xing))。也就是(shi)說在泛(fan)(fan)(fan)型(xing)(xing)(xing)使用(yong)過程中,

操作(zuo)的(de)數據類(lei)型(xing)被(bei)指定為(wei)一(yi)個參(can)數,這(zhe)種參(can)數類(lei)型(xing)可以(yi)用在類(lei)、接口(kou)和方法中,分別被(bei)稱為(wei)泛(fan)型(xing)類(lei)、泛(fan)型(xing)接口(kou)、泛(fan)型(xing)方法。

2. 一個栗子

一個(ge)被舉了(le)無(wu)數次的例子:

List arrayList = new ArrayList();
arrayList.add("aaaa");
arrayList.add(100);

for(int i = 0; i< arrayList.size();i++){
    String item = (String)arrayList.get(i);
    Log.d("泛型測試","item = " + item);
}

毫無疑問,程序的運(yun)行(xing)結果(guo)會以(yi)崩(beng)潰結束:

java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.String

ArrayList可以存放任意類(lei)(lei)型(xing),例子(zi)中添加了(le)(le)一個(ge)String類(lei)(lei)型(xing),添加了(le)(le)一個(ge)Integer類(lei)(lei)型(xing),再使用(yong)時都(dou)以String的方式使用(yong),因此程序崩潰(kui)了(le)(le)。為了(le)(le)解(jie)決類(lei)(lei)似這(zhe)樣的問題(ti)(在編譯階段就可以解(jie)決),泛型(xing)應(ying)運而(er)生。

我(wo)們將第(di)一行聲明初始化(hua)list的代碼更改一下(xia),編(bian)譯(yi)(yi)器會在編(bian)譯(yi)(yi)階段就能(neng)夠(gou)幫(bang)我(wo)們發現類似這樣的問題。

List<String> arrayList = new ArrayList<String>();
...
//arrayList.add(100); 在編(bian)譯階段(duan),編(bian)譯器就會(hui)報錯

3. 特性

泛型只在編譯階段有效。看下面的代碼:

List<String> stringArrayList = new ArrayList<String>();
List<Integer> integerArrayList = new ArrayList<Integer>();

Class classStringArrayList = stringArrayList.getClass();
Class classIntegerArrayList = integerArrayList.getClass();

if(classStringArrayList.equals(classIntegerArrayList)){
    Log.d("泛型測試","類型相同");
}

輸出結果:D/泛型測試: 類型相同

通過上面(mian)的例子可以證明(ming),在(zai)編(bian)譯(yi)(yi)之后程(cheng)序會(hui)采取去泛(fan)(fan)型(xing)(xing)化(hua)的措施(shi)。也就(jiu)是(shi)說Java中(zhong)的泛(fan)(fan)型(xing)(xing),只在(zai)編(bian)譯(yi)(yi)階(jie)段有效。在(zai)編(bian)譯(yi)(yi)過程(cheng)中(zhong),正確(que)檢(jian)驗泛(fan)(fan)型(xing)(xing)結果后,會(hui)將泛(fan)(fan)型(xing)(xing)的相(xiang)關信息擦出,并且在(zai)對象進入(ru)和離開方法(fa)的邊界處添加類型(xing)(xing)檢(jian)查和類型(xing)(xing)轉換的方法(fa)。也就(jiu)是(shi)說,泛(fan)(fan)型(xing)(xing)信息不會(hui)進入(ru)到運行時階(jie)段。

對此總(zong)結成一句話:泛型(xing)(xing)類(lei)型(xing)(xing)在邏輯上(shang)(shang)看以看成是(shi)多個不同的類(lei)型(xing)(xing),實(shi)際上(shang)(shang)都是(shi)相同的基本類(lei)型(xing)(xing)。

4. 泛型的使用

泛(fan)型有三種使用方式,分(fen)別為:泛(fan)型類(lei)、泛(fan)型接口、泛(fan)型方法

4.3 泛型類

泛型類(lei)型用于類(lei)的(de)(de)(de)定義中,被稱為泛型類(lei)。通過泛型可以完成對(dui)(dui)一組類(lei)的(de)(de)(de)操作對(dui)(dui)外開放相同的(de)(de)(de)接口。最典型的(de)(de)(de)就是各種(zhong)容器類(lei),如(ru):List、Set、Map。

泛型類的(de)最基本(ben)寫(xie)法(這么看可能會(hui)有(you)點暈,會(hui)在下面的(de)例子(zi)中詳解):

class 類名稱 <泛型標識:可以隨便寫任意標識號,標識指定的泛型的類型>{
  private 泛型標識 /*(成(cheng)員變量(liang)類(lei)型)*/ var; 
  .....

  }
}

一個最(zui)普(pu)通的泛型類(lei):

//此處T可以隨便寫(xie)為任意標識(shi),常(chang)見的如T、E、K、V等形式的參數常(chang)用于(yu)表示(shi)泛型
//在實(shi)例化泛型類時,必須(xu)指定T的具(ju)體類型
public class Generic<T>{ 
    //key這個成員變量的類(lei)(lei)型為T,T的類(lei)(lei)型由外部指(zhi)定(ding)  
    private T key;

    public Generic(T key) { //泛型(xing)(xing)構造方法(fa)形參(can)key的(de)(de)類(lei)型(xing)(xing)也為T,T的(de)(de)類(lei)型(xing)(xing)由外部指定
        this.key = key;
    }

    public T getKey(){ //泛(fan)型(xing)方法getKey的(de)(de)返(fan)回值(zhi)類型(xing)為T,T的(de)(de)類型(xing)由外部指(zhi)定(ding)
        return key;
    }
}

 

//泛型的(de)類(lei)型參數(shu)只能是(shi)類(lei)類(lei)型(包括自(zi)定義類(lei)),不能是(shi)簡單類(lei)型
//傳入的實參類型需與泛型的類型參數(shu)類型相(xiang)同,即為(wei)Integer.
Generic<Integer> genericInteger = new Generic<Integer>(123456);

//傳入的(de)實參(can)類(lei)型需與泛型的(de)類(lei)型參(can)數類(lei)型相同,即為String.
Generic<String> genericString = new Generic<String>("key_vlaue");
Log.d("泛型測試","key is " + genericInteger.getKey());
Log.d("泛型測試","key is " + genericString.getKey());

 

12-27 09:20:04.432 13063-13063/? D/泛型測(ce)試: key is 123456
12-27 09:20:04.432 13063-13063/? D/泛型測(ce)試: key is key_vlaue

定義的泛(fan)(fan)型(xing)類(lei)(lei),就一(yi)定要傳(chuan)入(ru)泛(fan)(fan)型(xing)類(lei)(lei)型(xing)實參(can)么?并(bing)不是這樣,在(zai)使用泛(fan)(fan)型(xing)的時候如果(guo)傳(chuan)入(ru)泛(fan)(fan)型(xing)實參(can),則會根據傳(chuan)入(ru)的泛(fan)(fan)型(xing)實參(can)做相應(ying)的限制(zhi)(zhi),此(ci)時泛(fan)(fan)型(xing)才(cai)會起到(dao)本(ben)應(ying)起到(dao)的限制(zhi)(zhi)作用。如果(guo)不傳(chuan)入(ru)泛(fan)(fan)型(xing)類(lei)(lei)型(xing)實參(can)的話(hua),在(zai)泛(fan)(fan)型(xing)類(lei)(lei)中使用泛(fan)(fan)型(xing)的方法或(huo)成員(yuan)變量定義的類(lei)(lei)型(xing)可以為任何的類(lei)(lei)型(xing)。

看一個例子:

Generic generic = new Generic("111111");
Generic generic1 = new Generic(4444);
Generic generic2 = new Generic(55.55);
Generic generic3 = new Generic(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 111111
D/泛型測試: key is 4444
D/泛型測試: key is 55.55
D/泛型測試: key is false

 

注意:

  • 泛型(xing)的類型(xing)參數只能是類類型(xing),不(bu)能是簡單類型(xing)。
  • 不能對(dui)確切的泛型(xing)(xing)類型(xing)(xing)使用instanceof操(cao)作(zuo)。如下(xia)面的操(cao)作(zuo)是非(fei)法的,編譯時會出錯。

  if(ex_num instanceof Generic<Number>){ }

4.4 泛型接口

泛(fan)型(xing)接口與泛(fan)型(xing)類(lei)的定義(yi)及使用(yong)基本相同。泛(fan)型(xing)接口常被用(yong)在各(ge)種類(lei)的生產器中,可以看一(yi)個(ge)例子:

//定義一個泛型接口(kou)
public interface Generator<T> {
    public T next();
}

當(dang)實(shi)現泛型接口(kou)的類(lei),未傳(chuan)入泛型實(shi)參(can)時:

/**
 * 未傳入泛(fan)型(xing)實參(can)時,與泛(fan)型(xing)類的(de)定義相同,在(zai)聲(sheng)(sheng)明(ming)類的(de)時候(hou),需將泛(fan)型(xing)的(de)聲(sheng)(sheng)明(ming)也一起加到類中(zhong)
 * 即(ji):class FruitGenerator<T> implements Generator<T>{
 * 如(ru)果不(bu)聲(sheng)(sheng)明(ming)泛(fan)型(xing),如(ru):class FruitGenerator implements Generator<T>,編譯器(qi)會(hui)報錯:"Unknown class"
 */
class FruitGenerator<T> implements Generator<T>{
    @Override
    public T next() {
        return null;
    }
}

當實(shi)現(xian)泛(fan)型接口(kou)的類,傳入泛(fan)型實(shi)參(can)時:

/**
 * 傳(chuan)入(ru)(ru)泛(fan)型(xing)(xing)(xing)實(shi)參(can)時(shi):
 * 定義一個(ge)生產器(qi)實(shi)現(xian)這個(ge)接(jie)口(kou)(kou),雖然我們只(zhi)創建了一個(ge)泛(fan)型(xing)(xing)(xing)接(jie)口(kou)(kou)Generator<T>
 * 但(dan)是我們可以為T傳(chuan)入(ru)(ru)無數個(ge)實(shi)參(can),形成(cheng)無數種類(lei)型(xing)(xing)(xing)的(de)(de)(de)Generator接(jie)口(kou)(kou)。
 * 在實(shi)現(xian)類(lei)實(shi)現(xian)泛(fan)型(xing)(xing)(xing)接(jie)口(kou)(kou)時(shi),如(ru)已將泛(fan)型(xing)(xing)(xing)類(lei)型(xing)(xing)(xing)傳(chuan)入(ru)(ru)實(shi)參(can)類(lei)型(xing)(xing)(xing),則所有使用泛(fan)型(xing)(xing)(xing)的(de)(de)(de)地方都(dou)要替換(huan)成(cheng)傳(chuan)入(ru)(ru)的(de)(de)(de)實(shi)參(can)類(lei)型(xing)(xing)(xing)
 * 即:Generator<T>,public T next();中的(de)(de)(de)的(de)(de)(de)T都(dou)要替換(huan)成(cheng)傳(chuan)入(ru)(ru)的(de)(de)(de)String類(lei)型(xing)(xing)(xing)。
 */
public class FruitGenerator implements Generator<String> {

    private String[] fruits = new String[]{"Apple", "Banana", "Pear"};

    @Override
    public String next() {
        Random rand = new Random();
        return fruits[rand.nextInt(3)];
    }
}

4.5 泛型通配符

我們知道IngeterNumber的一個子類,同時在特性章節中我們也驗證過Generic<Ingeter>Generic<Number>實際上是相同的一種基本類型。那么問題來了,在使用Generic<Number>作為形參的方法中,能否使用Generic<Ingeter>的實例傳入呢?在邏輯上類似于Generic<Number>Generic<Ingeter>是否可以看成具有父子關系的泛型類型呢?

為了弄清楚這個問題,我們使用Generic<T>這個泛型類繼續看下面的例子:

public void showKeyValue1(Generic<Number> obj){
    Log.d("泛型測試","key value is " + obj.getKey());
}

 

Generic<Integer> gInteger = new Generic<Integer>(123);
Generic<Number> gNumber = new Generic<Number>(456);

showKeyValue(gNumber);

// showKeyValue這個方法編譯器會為我們報錯:Generic<java.lang.Integer> 
// cannot be applied to Generic<java.lang.Number>
// showKeyValue(gInteger);

 

通過提示信息我們可以看到Generic<Integer>不能被看作為`Generic<Number>的子類。由此可以看出:同一種泛型可以對應多個版本(因為參數類型是不確定的),不同版本的泛型類實例是不兼容的。

回到上面的例子,如何解決上面的問題?總不能為了定義一個新的方法來處理Generic<Integer>類型的類,這顯然與java中的多臺理念相違背。因此我們需要一個在邏輯上可以表示同時是Generic<Integer>Generic<Number>父類的引用類型。由此類型通配符應運而生。

我們(men)可以(yi)將上(shang)面(mian)的(de)方法(fa)改一下:

public void showKeyValue1(Generic<?> obj){
    Log.d("泛型測試","key value is " + obj.getKey());
}

類(lei)(lei)型(xing)(xing)(xing)(xing)通配符一般是(shi)使(shi)用?代(dai)替具(ju)體的(de)類(lei)(lei)型(xing)(xing)(xing)(xing)實(shi)(shi)參(can)(can)(can)(can),注(zhu)意了,此(ci)處’?’是(shi)類(lei)(lei)型(xing)(xing)(xing)(xing)實(shi)(shi)參(can)(can)(can)(can),而不是(shi)類(lei)(lei)型(xing)(xing)(xing)(xing)形(xing)參(can)(can)(can)(can) 。重要(yao)說三遍!此(ci)處’?’是(shi)類(lei)(lei)型(xing)(xing)(xing)(xing)實(shi)(shi)參(can)(can)(can)(can),而不是(shi)類(lei)(lei)型(xing)(xing)(xing)(xing)形(xing)參(can)(can)(can)(can) ! 此(ci)處’?’是(shi)類(lei)(lei)型(xing)(xing)(xing)(xing)實(shi)(shi)參(can)(can)(can)(can),而不是(shi)類(lei)(lei)型(xing)(xing)(xing)(xing)形(xing)參(can)(can)(can)(can) !再直白點的(de)意思就(jiu)是(shi),此(ci)處的(de)?和Number、String、Integer一樣都是(shi)一種實(shi)(shi)際的(de)類(lei)(lei)型(xing)(xing)(xing)(xing),可以(yi)把?看成所有類(lei)(lei)型(xing)(xing)(xing)(xing)的(de)父類(lei)(lei)。是(shi)一種真實(shi)(shi)的(de)類(lei)(lei)型(xing)(xing)(xing)(xing)。

可(ke)以解決當具體類型(xing)(xing)不確(que)定的時(shi)候,這個通配符就是 ?&nbsp; ;當操作類型(xing)(xing)時(shi),不需要使用(yong)(yong)類型(xing)(xing)的具體功(gong)能(neng)時(shi),只使用(yong)(yong)Object類中的功(gong)能(neng)。那(nei)么可(ke)以用(yong)(yong) ? 通配符來(lai)表未知(zhi)類型(xing)(xing)。

4.6 泛型方法

在java中(zhong),泛型類的定義非常簡(jian)單,但是泛型方法就(jiu)比較復雜(za)了。

尤其是我們見(jian)到的大(da)多數泛型(xing)類中的成員方(fang)法(fa)也(ye)都使用了泛型(xing),有(you)的甚至泛型(xing)類中也(ye)包含(han)著泛型(xing)方(fang)法(fa),這樣在(zai)初學者中非常容(rong)易將泛型(xing)方(fang)法(fa)理解錯(cuo)了。

泛(fan)型(xing)(xing)類(lei),是(shi)在實例(li)化類(lei)的(de)時候(hou)指明泛(fan)型(xing)(xing)的(de)具(ju)體類(lei)型(xing)(xing);泛(fan)型(xing)(xing)方(fang)法,是(shi)在調用方(fang)法的(de)時候(hou)指明泛(fan)型(xing)(xing)的(de)具(ju)體類(lei)型(xing)(xing) 。

/**
 * 泛(fan)型方法的基本(ben)介紹
 * @param tClass 傳入的泛型實參
 * @return T 返回(hui)值為(wei)(wei)T類(lei)(lei)(lei)型(xing)(xing)
 * 說明(ming)(ming):
 *     1)public 與(yu)(yu) 返回(hui)值中(zhong)(zhong)間<T>非常(chang)重(zhong)要(yao),可(ke)以(yi)理解為(wei)(wei)聲明(ming)(ming)此(ci)方(fang)法為(wei)(wei)泛(fan)(fan)(fan)型(xing)(xing)方(fang)法。
 *     2)只有聲明(ming)(ming)了<T>的(de)方(fang)法才(cai)是(shi)泛(fan)(fan)(fan)型(xing)(xing)方(fang)法,泛(fan)(fan)(fan)型(xing)(xing)類(lei)(lei)(lei)中(zhong)(zhong)的(de)使用了泛(fan)(fan)(fan)型(xing)(xing)的(de)成員方(fang)法并不是(shi)泛(fan)(fan)(fan)型(xing)(xing)方(fang)法。
 *     3)<T>表明(ming)(ming)該方(fang)法將使用泛(fan)(fan)(fan)型(xing)(xing)類(lei)(lei)(lei)型(xing)(xing)T,此(ci)時(shi)才(cai)可(ke)以(yi)在方(fang)法中(zhong)(zhong)使用泛(fan)(fan)(fan)型(xing)(xing)類(lei)(lei)(lei)型(xing)(xing)T。
 *     4)與(yu)(yu)泛(fan)(fan)(fan)型(xing)(xing)類(lei)(lei)(lei)的(de)定義一樣,此(ci)處T可(ke)以(yi)隨(sui)便寫為(wei)(wei)任意標識,常(chang)見的(de)如T、E、K、V等(deng)形式的(de)參數(shu)常(chang)用于(yu)表示泛(fan)(fan)(fan)型(xing)(xing)。
 */
public <T> T genericMethod(Class<T> tClass)throws InstantiationException ,
  IllegalAccessException{
        T instance = tClass.newInstance();
        return instance;
}

 

Object obj = genericMethod(Class.forName("com.test.test"));

4.6.1 泛型方法的基本用法

光(guang)看上面的(de)(de)例(li)子(zi)有的(de)(de)同學可能(neng)依然會非常迷糊,我們再(zai)通(tong)過一(yi)個(ge)例(li)子(zi),把我泛型方(fang)法再(zai)總結一(yi)下。

public class GenericTest {
   //這個(ge)類(lei)是個(ge)泛型(xing)類(lei),在上面已(yi)經介紹過
   public class Generic<T>{     
        private T key;

        public Generic(T key) {
            this.key = key;
        }

        //我想說的其實是(shi)(shi)這(zhe)個,雖然在方(fang)法中使用了泛型,但是(shi)(shi)這(zhe)并不(bu)是(shi)(shi)一(yi)個泛型方(fang)法。
        //這只是類中一個普通的(de)成員方法,只不(bu)過他的(de)返回(hui)值是在聲(sheng)明泛型類已經(jing)聲(sheng)明過的(de)泛型。
        //所以在(zai)這個方(fang)法(fa)中才可以繼續使用(yong) T 這個泛型。
        public T getKey(){
            return key;
        }

        /**
         * 這個方法顯然是有問題的(de),在編譯器會給我(wo)們(men)提(ti)示這樣(yang)的(de)錯誤信息(xi)"cannot reslove symbol E"
         * 因為在類的(de)聲明(ming)中(zhong)并未聲明(ming)泛型E,所(suo)以在使用(yong)E做形參和返回值類型時,編譯器會無法識(shi)別(bie)。
        public E setKey(E key){
             this.key = keu
        }
        */
    }

    /** 
     * 這(zhe)(zhe)(zhe)才(cai)是一(yi)個(ge)真正的泛(fan)(fan)(fan)型(xing)方法。
     * 首先在public與返回(hui)值之間的<T>必不(bu)可少,這(zhe)(zhe)(zhe)表明(ming)這(zhe)(zhe)(zhe)是一(yi)個(ge)泛(fan)(fan)(fan)型(xing)方法,并且聲明(ming)了一(yi)個(ge)泛(fan)(fan)(fan)型(xing)T
     * 這(zhe)(zhe)(zhe)個(ge)T可以出現在這(zhe)(zhe)(zhe)個(ge)泛(fan)(fan)(fan)型(xing)方法的任意位置.
     * 泛(fan)(fan)(fan)型(xing)的數量也可以為任意多個(ge) 
     *    如:public <T,K> K showKeyName(Generic<T> container){
     *        ...
     *        }
     */
    public <T> T showKeyName(Generic<T> container){
        System.out.println("container key :" + container.getKey());
        //當然這個例(li)子舉的不太合適,只是為了(le)說明泛型(xing)方法的特性(xing)。
        T test = container.getKey();
        return test;
    }

    //這也不是(shi)(shi)一個泛型方法(fa),這就是(shi)(shi)一個普(pu)通(tong)的方法(fa),只是(shi)(shi)使用了Generic<Number>這個泛型類做形參而已。
    public void showKeyValue1(Generic<Number> obj){
        Log.d("泛型測試","key value is " + obj.getKey());
    }

    //這(zhe)也(ye)不是一個泛型方法,這(zhe)也(ye)是一個普(pu)通(tong)的方法,只不過使用了泛型通(tong)配符?
    //同時這也印證了泛型通配符(fu)章節所(suo)描述的,?是(shi)一(yi)種類(lei)型實參,可以看做為Number等所(suo)有類(lei)的父類(lei)
    public void showKeyValue2(Generic<?> obj){
        Log.d("泛型測試","key value is " + obj.getKey());
    }

     /**
     * 這個(ge)方(fang)法是有問題的(de),編譯器會(hui)為我(wo)們(men)(men)提示錯誤信息:"UnKnown class 'E' "
     * 雖然(ran)我(wo)們(men)(men)聲(sheng)(sheng)明(ming)了<T>,也表明(ming)了這是一個(ge)可以處理泛(fan)型(xing)(xing)的(de)類型(xing)(xing)的(de)泛(fan)型(xing)(xing)方(fang)法。
     * 但是只聲(sheng)(sheng)明(ming)了泛(fan)型(xing)(xing)類型(xing)(xing)T,并未聲(sheng)(sheng)明(ming)泛(fan)型(xing)(xing)類型(xing)(xing)E,因此(ci)編譯器并不(bu)知(zhi)道該如何(he)處理E這個(ge)類型(xing)(xing)。
    public <T> T showKeyName(Generic<E> container){
        ...
    }  
    */

    /**
     * 這個方(fang)法(fa)也是有問題的,編(bian)譯(yi)器會為我們提示(shi)錯(cuo)誤信息:"UnKnown class 'T' "
     * 對于編(bian)譯(yi)器來說T這個類(lei)型(xing)并未項目中聲(sheng)(sheng)明(ming)過,因(yin)此編(bian)譯(yi)也不知道該如何編(bian)譯(yi)這個類(lei)。
     * 所以(yi)這也不是一(yi)個正確的泛型(xing)方(fang)法(fa)聲(sheng)(sheng)明(ming)。
    public void showkey(T genericObj){

    }
    */

    public static void main(String[] args) {


    }
}

4.6.2 類中的泛型方法

當然這并(bing)不是(shi)泛(fan)型方(fang)法的全部,泛(fan)型方(fang)法可以出現(xian)雜任何地方(fang)和(he)任何場景中(zhong)使用。但是(shi)有一(yi)種情況是(shi)非常特殊的,當泛(fan)型方(fang)法出現(xian)在泛(fan)型類(lei)中(zhong)時,我(wo)們再(zai)通過(guo)一(yi)個例子看(kan)一(yi)下

public class GenericFruit {
    class Fruit{
        @Override
        public String toString() {
            return "fruit";
        }
    }

    class Apple extends Fruit{
        @Override
        public String toString() {
            return "apple";
        }
    }

    class Person{
        @Override
        public String toString() {
            return "Person";
        }
    }

    class GenerateTest<T>{
        public void show_1(T t){
            System.out.println(t.toString());
        }

        //在泛(fan)型(xing)類(lei)(lei)中(zhong)聲(sheng)明了一個泛(fan)型(xing)方法,使用泛(fan)型(xing)E,這種泛(fan)型(xing)E可(ke)以(yi)為任意類(lei)(lei)型(xing)。可(ke)以(yi)類(lei)(lei)型(xing)與T相同,也可(ke)以(yi)不同。
        //由于泛(fan)(fan)型(xing)(xing)方法在(zai)聲(sheng)明的時候會聲(sheng)明泛(fan)(fan)型(xing)(xing)<E>,因此(ci)即使在(zai)泛(fan)(fan)型(xing)(xing)類中并未聲(sheng)明泛(fan)(fan)型(xing)(xing),編譯(yi)器也能夠正確識(shi)別泛(fan)(fan)型(xing)(xing)方法中識(shi)別的泛(fan)(fan)型(xing)(xing)。
        public <E> void show_3(E t){
            System.out.println(t.toString());
        }

        //在泛型(xing)類(lei)中聲明(ming)了一(yi)(yi)個泛型(xing)方法,使用(yong)泛型(xing)T,注意(yi)這(zhe)個T是(shi)一(yi)(yi)種全新的類(lei)型(xing),可以與(yu)泛型(xing)類(lei)中聲明(ming)的T不是(shi)同一(yi)(yi)種類(lei)型(xing)。
        public <T> void show_2(T t){
            System.out.println(t.toString());
        }
    }

    public static void main(String[] args) {
        Apple apple = new Apple();
        Person person = new Person();

        GenerateTest<Fruit> generateTest = new GenerateTest<Fruit>();
        //apple是(shi)Fruit的子類,所以這里(li)可(ke)以
        generateTest.show_1(apple);
        //編(bian)譯器(qi)會報錯(cuo),因為泛型類型實(shi)參指(zhi)定的是Fruit,而傳入的實(shi)參類是Person
        //generateTest.show_1(person);

        //使(shi)用這兩個方法都(dou)可以(yi)成功
        generateTest.show_2(apple);
        generateTest.show_2(person);

        //使用這兩個方法(fa)也都(dou)可以成功
        generateTest.show_3(apple);
        generateTest.show_3(person);
    }
}

4.6.3 泛型方法與可變參數

再看(kan)一個泛型方法和(he)可變參數的例子:

public <T> void printMsg( T... args){
    for(T t : args){
        Log.d("泛型測試","t is " + t);
    }
}
printMsg("111",222,"aaaa","2323.4",55.55);

4.6.4 靜態方法與泛型

靜(jing)態方(fang)法(fa)有(you)一(yi)種情況需要注意(yi)一(yi)下,那(nei)就(jiu)是在類(lei)(lei)中的靜(jing)態方(fang)法(fa)使用泛(fan)型(xing):靜(jing)態方(fang)法(fa)無法(fa)訪問(wen)類(lei)(lei)上定(ding)義的泛(fan)型(xing);如果靜(jing)態方(fang)法(fa)操作的引用數據類(lei)(lei)型(xing)不(bu)確定(ding)的時候(hou),必須要將泛(fan)型(xing)定(ding)義在方(fang)法(fa)上。

即(ji):如果(guo)靜態方法要使用泛型的(de)話,必須(xu)將(jiang)靜態方法也定義成泛型方法 ;。

public class StaticGenerator<T> {
    ....
    ....
    /**
     * 如果在類中定義使用(yong)泛(fan)(fan)(fan)型(xing)的(de)靜(jing)態方(fang)法(fa),需要添加(jia)額外的(de)泛(fan)(fan)(fan)型(xing)聲(sheng)明(將這個方(fang)法(fa)定義成泛(fan)(fan)(fan)型(xing)方(fang)法(fa))
     * 即使靜(jing)態方(fang)法(fa)要使用(yong)泛(fan)(fan)(fan)型(xing)類中已經聲(sheng)明過的(de)泛(fan)(fan)(fan)型(xing)也(ye)不(bu)可以(yi)。
     * 如:public static void show(T t){..},此時編譯器(qi)會(hui)提示錯(cuo)誤信息:
          "StaticGenerator cannot be refrenced from static context"
     */
    public static <T> void show(T t){

    }
}

4.6.5 泛型方法總結

泛型方(fang)(fang)法能使方(fang)(fang)法獨立于類而(er)產(chan)生變化,以下是一個基本的指導原則:

無論何時(shi),如果你(ni)能做到,你(ni)就(jiu)(jiu)該盡(jin)量使(shi)用泛(fan)型(xing)方法(fa)。也就(jiu)(jiu)是說,如果使(shi)用泛(fan)型(xing)方法(fa)將整個(ge)類(lei)泛(fan)型(xing)化,

那么就(jiu)應該使用泛(fan)型方(fang)法(fa)(fa)。另外對于一個static的(de)方(fang)法(fa)(fa)而已(yi),無法(fa)(fa)訪問泛(fan)型類型的(de)參數。

所以如果static方(fang)法(fa)要使(shi)用泛(fan)型能(neng)力,就必須使(shi)其成為泛(fan)型方(fang)法(fa)。

4.6 泛型上下邊界

在使用泛型(xing)的時候,我們還可以為(wei)傳入的泛型(xing)類(lei)(lei)型(xing)實(shi)參(can)進行上下邊(bian)界的限(xian)制(zhi),如:類(lei)(lei)型(xing)實(shi)參(can)只準傳入某種類(lei)(lei)型(xing)的父類(lei)(lei)或某種類(lei)(lei)型(xing)的子(zi)類(lei)(lei)。

為泛型(xing)(xing)添加上邊界,即傳入的類型(xing)(xing)實參必須是指定(ding)類型(xing)(xing)的子類型(xing)(xing)

public void showKeyValue1(Generic<? extends Number> obj){
    Log.d("泛型測試","key value is " + obj.getKey());
}
Generic<String> generic1 = new Generic<String>("11111");
Generic<Integer> generic2 = new Generic<Integer>(2222);
Generic<Float> generic3 = new Generic<Float>(2.4f);
Generic<Double> generic4 = new Generic<Double>(2.56);

//這一行代(dai)碼(ma)編譯器會(hui)提示錯誤,因(yin)為(wei)String類(lei)(lei)型并(bing)不是(shi)Number類(lei)(lei)型的子類(lei)(lei)
//showKeyValue1(generic1);

showKeyValue1(generic2);
showKeyValue1(generic3);
showKeyValue1(generic4);

如果我們(men)把泛型類的定義也改(gai)一下:

public class Generic<T extends Number>{
    private T key;

    public Generic(T key) {
        this.key = key;
    }

    public T getKey(){
        return key;
    }
}

 

//這一(yi)行代碼也會報錯,因為String不是Number的子類(lei)
Generic<String> generic1 = new Generic<String>("11111");

再(zai)來一個泛(fan)型方法的(de)例子:

 

//在(zai)(zai)泛型(xing)方(fang)法中添加上下邊界(jie)限制的(de)(de)時(shi)候,必須在(zai)(zai)權限聲明(ming)(ming)與返回(hui)值之間(jian)的(de)(de)<T>上添加上下邊界(jie),即(ji)在(zai)(zai)泛型(xing)聲明(ming)(ming)的(de)(de)時(shi)候添加
//public <T> T showKeyName(Generic<T extends Number> container),編譯器會報錯:"Unexpected bound"
public <T extends Number> T showKeyName(Generic<T> container){
    System.out.println("container key :" + container.getKey());
    T test = container.getKey();
    return test;
}

通(tong)過上(shang)面(mian)的(de)兩個例子(zi)可以看出:泛(fan)型的(de)上(shang)下邊(bian)界(jie)添加,必須與泛(fan)型的(de)聲(sheng)明在(zai)一起 。

4.7 關于泛型數組要提一下

看到(dao)了很多文章(zhang)中都(dou)會(hui)提起泛型(xing)(xing)數(shu)組,經過(guo)查看sun的(de)(de)說明文檔,在java中是”不能創建一個確切的(de)(de)泛型(xing)(xing)類型(xing)(xing)的(de)(de)數(shu)組”的(de)(de)。

也就(jiu)是說下面的(de)這個例(li)子(zi)是不可以(yi)的(de):

List<String>[] ls = new ArrayList<String>[10];  

而使用通配符創(chuang)建泛(fan)型數組(zu)是可以的,如下面(mian)這個例(li)子:

List<?>[] ls = new ArrayList<?>[10]; 

這樣(yang)也是可以的:

List<String>[] ls = new ArrayList[10];

下面(mian)使用的一個(ge)例子來(lai)說(shuo)明這個(ge)問題(ti):

List<String>[] lsa = new List<String>[10]; // Not really allowed.    
Object o = lsa;    
Object[] oa = (Object[]) o;    
List<Integer> li = new ArrayList<Integer>();    
li.add(new Integer(3));    
oa[1] = li; // Unsound, but passes run time store check    
String s = lsa[1].get(0); // Run-time error: ClassCastException.

 

這種情況下,由于JVM泛型的擦除機制,在運行時JVM是不知道泛型信息的,所以可以給oa[1]賦上一個ArrayList而不會出現異常,

但是在取出數據的時候卻要做一次類型轉換,所以就會出現ClassCastException,如果可以進行泛型數組的聲明,

上面說的這種情況在編譯期將不會出現任何的警告和錯誤,只有在運行時才會出錯。 而對泛型數組的聲明進行限制,對于這樣的情況,可以在編譯期提示代碼有類型安全問題,比沒有任何提示要強很多。

 

下面采用通(tong)配符的(de)方式是(shi)被允許的(de):數組的(de)類型不可(ke)以是(shi)類型變量,除非(fei)是(shi)采用通(tong)配符的(de)方式,因(yin)為對于通(tong)配符的(de)方式,最后(hou)取出數據是(shi)要做顯(xian)式的(de)類型轉換(huan)的(de)。

List<?>[] lsa = new List<?>[10]; // OK, array of unbounded wildcard type.    
Object o = lsa;    
Object[] oa = (Object[]) o;    
List<Integer> li = new ArrayList<Integer>();    
li.add(new Integer(3));    
oa[1] = li; // Correct.    
Integer i = (Integer) lsa[1].get(0); // OK 

5. 最后

本(ben)文中的例子主(zhu)要是為了(le)闡述(shu)泛(fan)型(xing)中的一(yi)(yi)些思想(xiang)而(er)簡單舉出(chu)的,并不一(yi)(yi)定(ding)有著實際(ji)的可(ke)用(yong)性。另外,一(yi)(yi)提(ti)到(dao)泛(fan)型(xing),相信大家(jia)用(yong)到(dao)最多的就是在集(ji)合中,其實,在實際(ji)的編程過程中,自(zi)己可(ke)以使用(yong)泛(fan)型(xing)去簡化開發,且(qie)能很好的保證代碼質量。

  
posted on 2018-03-19 17:28  little fat  閱讀(740773)  評論(36)    收藏  舉報