0x6 Java系列Java 泛型詳解-絕對(duì)是對(duì)泛型方法講解最詳細(xì)的,沒有之一

      網(wǎng)友投稿 676 2025-04-02

      概述

      泛型在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ù)類型可以用在類、接口和方法中,分別被稱為泛型類、泛型接口、泛型方法。

      一個(gè)栗子

      一個(gè)被舉了無數(shù)次的例子:

      1.??ListarrayList?=newArrayList();

      2.??arrayList.add("aaaa");

      3.??arrayList.add(100);

      4.

      5.??for(int?i?=0;i

      6.???? ?Stringitem?=(String)arrayList.get(i);

      7.???? ?Log.d("泛型測(cè)試","item = "+item);

      8.??}

      毫無疑問,程序的運(yùn)行結(jié)果會(huì)以崩潰結(jié)束:

      1.??java.lang.ClassCastException:java.lang.Integercannot be cast to java.lang.String

      ArrayList可以存放任意類型,例子中添加了一個(gè)String類型,添加了一個(gè)Integer類型,再使用時(shí)都以String的方式使用,因此程序崩潰了。為了解決類似這樣的問題(在編譯階段就可以解決),泛型應(yīng)運(yùn)而生。

      我們將第一行聲明初始化list的代碼更改一下,編譯器會(huì)在編譯階段就能夠幫我們發(fā)現(xiàn)類似這樣的問題。

      1.??ListarrayList?=newArrayList();

      2.??...

      3.??//arrayList.add(100);?在編譯階段,編譯器就會(huì)報(bào)錯(cuò)

      特性

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

      1.??ListstringArrayList?=newArrayList();

      2.??ListintegerArrayList?=newArrayList();

      3.

      4.??ClassclassStringArrayList?=stringArrayList.getClass();

      5.??ClassclassIntegerArrayList?=integerArrayList.getClass();

      6.

      7.??if(classStringArrayList.equals(classIntegerArrayList)){

      8.???? ?Log.d("泛型測(cè)試","類型相同");

      9.??}

      輸出結(jié)果:D/泛型測(cè)試: 類型相同。

      通過上面的例子可以證明,在編譯之后程序會(huì)采取去泛型化的措施。也就是說Java中的泛型,只在編譯階段有效。在編譯過程中,正確檢驗(yàn)泛型結(jié)果后,會(huì)將泛型的相關(guān)信息擦出,并且在對(duì)象進(jìn)入和離開方法的邊界處添加類型檢查和類型轉(zhuǎn)換的方法。也就是說,泛型信息不會(huì)進(jìn)入到運(yùn)行時(shí)階段。

      對(duì)此總結(jié)成一句話:泛型類型在邏輯上看以看成是多個(gè)不同的類型,實(shí)際上都是相同的基本類型。

      泛型的使用

      泛型有三種使用方式,分別為:泛型類、泛型接口、泛型方法

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

      泛型類的最基本寫法(這么看可能會(huì)有點(diǎn)暈,會(huì)在下面的例子中詳解):

      1.??class類名稱<泛型標(biāo)識(shí):可以隨便寫任意標(biāo)識(shí)號(hào),標(biāo)識(shí)指定的泛型的類型>{

      2.????private泛型標(biāo)識(shí)/*(成員變量類型)*/var;

      3.????.....

      4.

      5.????}

      6.??}

      一個(gè)最普通的泛型類:

      //此處T可以隨便寫為任意標(biāo)識(shí),常見的如T、E、K、V等形式的參數(shù)常用于表示泛型//在實(shí)例化泛型類時(shí),必須指定T的具體類型

      1.??publicclassGeneric{

      2.???? ?//key這個(gè)成員變量的類型為T,T的類型由外部指定

      3.???? ?privateT key;

      4.

      5.???? ?publicGeneric(T key){//泛型構(gòu)造方法形參key的類型也為T,T的類型由外部指定

      6.???? ? ? ?this.key?=key;

      7.???? ?}

      8.

      9.???? ?publicT getKey(){//泛型方法getKey的返回值類型為T,T的類型由外部指定

      10.???? ? ? ?returnkey;

      11.???? ?}

      12.??}

      1.??//泛型的類型參數(shù)只能是類類型(包括自定義類),不能是簡單類型

      2.??//傳入的實(shí)參類型需與泛型的類型參數(shù)類型相同,即為Integer.

      3.??GenericgenericInteger?=newGeneric(123456);

      4.

      5.??//傳入的實(shí)參類型需與泛型的類型參數(shù)類型相同,即為String.

      6.??GenericgenericString?=newGeneric("key_vlaue");

      7.??Log.d("泛型測(cè)試","key is "+genericInteger.getKey());

      8.??Log.d("泛型測(cè)試","key is "+genericString.getKey());

      1.??12-2709:20:04.43213063-13063/?D/泛型測(cè)試:key?is123456

      2.??12-2709:20:04.43213063-13063/?D/泛型測(cè)試:key?iskey_vlaue

      定義的泛型類,就一定要傳入泛型類型實(shí)參么?并不是這樣,在使用泛型的時(shí)候如果傳入泛型實(shí)參,則會(huì)根據(jù)傳入的泛型實(shí)參做相應(yīng)的限制,此時(shí)泛型才會(huì)起到本應(yīng)起到的限制作用。如果不傳入泛型類型實(shí)參的話,在泛型類中使用泛型的方法或成員變量定義的類型可以為任何的類型。

      看一個(gè)例子:

      1.??Genericgeneric?=newGeneric("111111");

      2.??Genericgeneric1?=newGeneric(4444);

      3.??Genericgeneric2?=newGeneric(55.55);

      4.??Genericgeneric3?=newGeneric(false);

      5.

      6.??Log.d("泛型測(cè)試","key is "+generic.getKey());

      7.??Log.d("泛型測(cè)試","key is "+generic1.getKey());

      8.??Log.d("泛型測(cè)試","key is "+generic2.getKey());

      9.??Log.d("泛型測(cè)試","key is "+generic3.getKey());

      1.??D/泛型測(cè)試:key?is111111

      2.??D/泛型測(cè)試:key?is4444

      3.??D/泛型測(cè)試:key?is55.55

      4.??D/泛型測(cè)試:key?isfalse

      注意:

      §?泛型的類型參數(shù)只能是類類型,不能是簡單類型。

      §?不能對(duì)確切的泛型類型使用instanceof操作。如下面的操作是非法的,編譯時(shí)會(huì)出錯(cuò)。

      1.??if(ex_num?instanceofGeneric){

      2.??}

      泛型接口

      泛型接口與泛型類的定義及使用基本相同。泛型接口常被用在各種類的生產(chǎn)器中,可以看一個(gè)例子:

      1.??//定義一個(gè)泛型接口

      2.??publicinterfaceGenerator{

      3.???? ?publicT next();

      4.??}

      當(dāng)實(shí)現(xiàn)泛型接口的類,未傳入泛型實(shí)參時(shí):

      1.??/**

      2.???*?未傳入泛型實(shí)參時(shí),與泛型類的定義相同,在聲明類的時(shí)候,需將泛型的聲明也一起加到類中

      3.???*?即:class FruitGenerator implements Generator{

      4.???*?如果不聲明泛型,如:class FruitGenerator implements Generator,編譯器會(huì)報(bào)錯(cuò):"Unknown class"

      5.???*/

      6.??classFruitGeneratorimplementsGenerator{

      7.???? ?@Override

      8.???? ?publicT next(){

      9.???? ? ? ?returnnull;

      10.???? ?}

      11.??}

      當(dāng)實(shí)現(xiàn)泛型接口的類,傳入泛型實(shí)參時(shí):

      1.??/**

      2.???*?傳入泛型實(shí)參時(shí):

      3.???*?定義一個(gè)生產(chǎn)器實(shí)現(xiàn)這個(gè)接口,雖然我們只創(chuàng)建了一個(gè)泛型接口Generator

      4.???*?但是我們可以為T傳入無數(shù)個(gè)實(shí)參,形成無數(shù)種類型的Generator接口。

      5.???*?在實(shí)現(xiàn)類實(shí)現(xiàn)泛型接口時(shí),如已將泛型類型傳入實(shí)參類型,則所有使用泛型的地方都要替換成傳入的實(shí)參類型

      6.???*?即:Generator,public T next();中的的T都要替換成傳入的String類型。

      7.???*/

      8.??publicclassFruitGeneratorimplementsGenerator{

      9.

      10.???? ?privateString[]fruits?=newString[]{"Apple","Banana","Pear"};

      11.

      12.???? ?@Override

      13.???? ?publicStringnext(){

      14.???? ? ? ?Randomrand?=newRandom();

      15.???? ? ? ?returnfruits[rand.nextInt(3)];

      16.???? ?}

      17.??}

      泛型通配符

      我們知道Ingeter是Number的一個(gè)子類,同時(shí)在特性章節(jié)中我們也驗(yàn)證過Generic與Generic實(shí)際上是相同的一種基本類型。那么問題來了,在使用Generic作為形參的方法中,能否使用Generic的實(shí)例傳入呢?在邏輯上類似于Generic和Generic是否可以看成具有父子關(guān)系的泛型類型呢?

      為了弄清楚這個(gè)問題,我們使用Generic這個(gè)泛型類繼續(xù)看下面的例子:

      1.??publicvoidshowKeyValue1(Genericobj){

      2.???? ?Log.d("泛型測(cè)試","key value is "+obj.getKey());

      3.??}

      1.??GenericgInteger?=newGeneric(123);

      2.??GenericgNumber?=newGeneric(456);

      3.

      4.??showKeyValue(gNumber);

      5.

      6.??// showKeyValue這個(gè)方法編譯器會(huì)為我們報(bào)錯(cuò):Generic

      7.??// cannot be applied to Generic

      8.??// showKeyValue(gInteger);

      通過提示信息我們可以看到Generic不能被看作為`Generic的子類。由此可以看出:同一種泛型可以對(duì)應(yīng)多個(gè)版本(因?yàn)閰?shù)類型是不確定的),不同版本的泛型類實(shí)例是不兼容的。

      回到上面的例子,如何解決上面的問題?總不能為了定義一個(gè)新的方法來處理Generic類型的類,這顯然與java中的多臺(tái)理念相違背。因此我們需要一個(gè)在邏輯上可以表示同時(shí)是Generic和Generic父類的引用類型。由此類型通配符應(yīng)運(yùn)而生。

      我們可以將上面的方法改一下:

      1.??publicvoidshowKeyValue1(Genericobj){

      2.???? ?Log.d("泛型測(cè)試","key value is "+obj.getKey());

      3.??}

      類型通配符一般是使用?代替具體的類型實(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í)候指明泛型的具體類型。

      1.??/**

      2.???*?泛型方法的基本介紹

      3.???* @param tClass?傳入的泛型實(shí)參

      4.???* @return T?返回值為T類型

      5.???*?說明:

      6.???* ? ? 1)public?與 返回值中間非常重要,可以理解為聲明此方法為泛型方法。

      7.???* ? ? 2)只有聲明了的方法才是泛型方法,泛型類中的使用了泛型的成員方法并不是泛型方法。

      8.???* ? ? 3)表明該方法將使用泛型類型T,此時(shí)才可以在方法中使用泛型類型T。

      9.???* ? ? 4)與泛型類的定義一樣,此處T可以隨便寫為任意標(biāo)識(shí),常見的如T、E、K、V等形式的參數(shù)常用于表示泛型。

      10.???*/

      11.??publicT genericMethod(ClasstClass)throwsInstantiationException,

      12.????IllegalAccessException{

      13.???? ? ? ?T instance?=tClass.newInstance();

      14.???? ? ? ?returninstance;

      15.??}

      1.??Objectobj?=genericMethod(Class.forName("com.test.test"));

      光看上面的例子有的同學(xué)可能依然會(huì)非常迷糊,我們?cè)偻ㄟ^一個(gè)例子,把我泛型方法再總結(jié)一下。

      1.??publicclassGenericTest{

      2.?????//這個(gè)類是個(gè)泛型類,在上面已經(jīng)介紹過

      3.?????publicclassGeneric{

      4.???? ? ? ?privateT key;

      5.

      6.???? ? ? ?publicGeneric(T key){

      7.???? ? ? ? ? ?this.key?=key;

      8.???? ? ? ?}

      9.

      10.???? ? ? ?//我想說的其實(shí)是這個(gè),雖然在方法中使用了泛型,但是這并不是一個(gè)泛型方法。

      11.???? ? ? ?//這只是類中一個(gè)普通的成員方法,只不過他的返回值是在聲明泛型類已經(jīng)聲明過的泛型。

      12.???? ? ? ?//所以在這個(gè)方法中才可以繼續(xù)使用T?這個(gè)泛型。

      13.???? ? ? ?publicT getKey(){

      14.???? ? ? ? ? ?returnkey;

      15.???? ? ? ?}

      16.

      17.???? ? ? ?/**

      18.???? ? ? ? *?這個(gè)方法顯然是有問題的,在編譯器會(huì)給我們提示這樣的錯(cuò)誤信息"cannot reslove symbol E"

      19.???? ? ? ? *?因?yàn)樵陬惖穆暶髦胁⑽绰暶鞣盒虴,所以在使用E做形參和返回值類型時(shí),編譯器會(huì)無法識(shí)別。

      20.???? ? ? ?public E setKey(E key){

      21.???? ? ? ? ? ? this.key = keu

      22.???? ? ? ?}

      23.???? ? ? ?*/

      24.???? ?}

      25.

      26.???? ?/**

      27.???? ? *?這才是一個(gè)真正的泛型方法。

      28.???? ? *?首先在public與返回值之間的必不可少,這表明這是一個(gè)泛型方法,并且聲明了一個(gè)泛型T

      29.???? ? *?這個(gè)T可以出現(xiàn)在這個(gè)泛型方法的任意位置.

      30.???? ? *?泛型的數(shù)量也可以為任意多個(gè)

      31.???? ? * ? ?如:public K showKeyName(Generic container){

      32.???? ? * ? ? ? ?...

      33.???? ? * ? ? ? ?}

      34.???? ? */

      35.???? ?publicT showKeyName(Genericcontainer){

      36.???? ? ? ?System.out.println("container key :"+container.getKey());

      37.???? ? ? ?//當(dāng)然這個(gè)例子舉的不太合適,只是為了說明泛型方法的特性。

      38.???? ? ? ?T test?=container.getKey();

      39.???? ? ? ?returntest;

      40.???? ?}

      41.

      42.???? ?//這也不是一個(gè)泛型方法,這就是一個(gè)普通的方法,只是使用了Generic這個(gè)泛型類做形參而已。

      43.???? ?publicvoidshowKeyValue1(Genericobj){

      44.???? ? ? ?Log.d("泛型測(cè)試","key value is "+obj.getKey());

      45.???? ?}

      46.

      47.???? ?//這也不是一個(gè)泛型方法,這也是一個(gè)普通的方法,只不過使用了泛型通配符?

      48.???? ?//同時(shí)這也印證了泛型通配符章節(jié)所描述的,?是一種類型實(shí)參,可以看做為Number等所有類的父類

      49.???? ?publicvoidshowKeyValue2(Genericobj){

      50.???? ? ? ?Log.d("泛型測(cè)試","key value is "+obj.getKey());

      51.???? ?}

      52.

      53.???? ??/**

      54.???? ? *?這個(gè)方法是有問題的,編譯器會(huì)為我們提示錯(cuò)誤信息:"UnKnown class 'E' "

      55.???? ? *?雖然我們聲明了,也表明了這是一個(gè)可以處理泛型的類型的泛型方法。

      56.???? ? *?但是只聲明了泛型類型T,并未聲明泛型類型E,因此編譯器并不知道該如何處理E這個(gè)類型。

      57.???? ?public T showKeyName(Generic container){

      58.???? ? ? ?...

      59.???? ?}

      60.???? ?*/

      61.

      62.???? ?/**

      63.???? ? *?這個(gè)方法也是有問題的,編譯器會(huì)為我們提示錯(cuò)誤信息:"UnKnown class 'T' "

      64.???? ? *?對(duì)于編譯器來說T這個(gè)類型并未項(xiàng)目中聲明過,因此編譯也不知道該如何編譯這個(gè)類。

      65.???? ? *?所以這也不是一個(gè)正確的泛型方法聲明。

      66.???? ?public void showkey(T genericObj){

      67.

      68.???? ?}

      69.???? ?*/

      70.

      71.???? ?publicstaticvoidmain(String[]args){

      72.

      73.

      74.???? ?}

      75.??}

      當(dāng)然這并不是泛型方法的全部,泛型方法可以出現(xiàn)雜任何地方和任何場(chǎng)景中使用。但是有一種情況是非常特殊的,當(dāng)泛型方法出現(xiàn)在泛型類中時(shí),我們?cè)偻ㄟ^一個(gè)例子看一下

      1.??publicclassGenericFruit{

      2.???? ?classFruit{

      3.???? ? ? ?@Override

      4.???? ? ? ?publicStringtoString(){

      5.???? ? ? ? ? ?return"fruit";

      6.???? ? ? ?}

      7.???? ?}

      8.

      9.???? ?classAppleextendsFruit{

      10.???? ? ? ?@Override

      11.???? ? ? ?publicStringtoString(){

      12.???? ? ? ? ? ?return"apple";

      13.???? ? ? ?}

      14.???? ?}

      15.

      16.???? ?classPerson{

      17.???? ? ? ?@Override

      18.???? ? ? ?publicStringtoString(){

      19.???? ? ? ? ? ?return"Person";

      20.???? ? ? ?}

      21.???? ?}

      22.

      23.???? ?classGenerateTest{

      24.???? ? ? ?publicvoidshow_1(T t){

      25.???? ? ? ? ? ?System.out.println(t.toString());

      26.???? ? ? ?}

      27.

      28.???? ? ? ?//在泛型類中聲明了一個(gè)泛型方法,使用泛型E,這種泛型E可以為任意類型。可以類型與T相同,也可以不同。

      29.???? ? ? ?//由于泛型方法在聲明的時(shí)候會(huì)聲明泛型,因此即使在泛型類中并未聲明泛型,編譯器也能夠正確識(shí)別泛型方法中識(shí)別的泛型。

      30.???? ? ? ?publicvoidshow_3(E t){

      31.???? ? ? ? ? ?System.out.println(t.toString());

      32.???? ? ? ?}

      33.

      34.???? ? ? ?//在泛型類中聲明了一個(gè)泛型方法,使用泛型T,注意這個(gè)T是一種全新的類型,可以與泛型類中聲明的T不是同一種類型。

      35.???? ? ? ?publicvoidshow_2(T t){

      36.???? ? ? ? ? ?System.out.println(t.toString());

      37.???? ? ? ?}

      38.???? ?}

      39.

      40.???? ?publicstaticvoidmain(String[]args){

      41.???? ? ? ?Appleapple?=newApple();

      42.???? ? ? ?Personperson?=newPerson();

      43.

      44.???? ? ? ?GenerateTestgenerateTest?=newGenerateTest();

      45.???? ? ? ?//apple是Fruit的子類,所以這里可以

      46.???? ? ? ?generateTest.show_1(apple);

      47.???? ? ? ?//編譯器會(huì)報(bào)錯(cuò),因?yàn)榉盒皖愋蛯?shí)參指定的是Fruit,而傳入的實(shí)參類是Person

      48.???? ? ? ?//generateTest.show_1(person);

      49.

      50.???? ? ? ?//使用這兩個(gè)方法都可以成功

      51.???? ? ? ?generateTest.show_2(apple);

      52.???? ? ? ?generateTest.show_2(person);

      53.

      54.???? ? ? ?//使用這兩個(gè)方法也都可以成功

      55.???? ? ? ?generateTest.show_3(apple);

      56.???? ? ? ?generateTest.show_3(person);

      57.???? ?}

      58.??}

      再看一個(gè)泛型方法和可變參數(shù)的例子:

      1.??publicvoidprintMsg(T...args){

      2.???? ?for(T t?:args){

      3.???? ? ? ?Log.d("泛型測(cè)試","t is "+t);

      4.???? ?}

      5.??}

      1.??printMsg("111",222,"aaaa","2323.4",55.55);

      靜態(tài)方法有一種情況需要注意一下,那就是在類中的靜態(tài)方法使用泛型:靜態(tài)方法無法訪問類上定義的泛型;如果靜態(tài)方法操作的引用數(shù)據(jù)類型不確定的時(shí)候,必須要將泛型定義在方法上。

      即:如果靜態(tài)方法要使用泛型的話,必須將靜態(tài)方法也定義成泛型方法。

      1.??publicclassStaticGenerator{

      2.???? ?....

      3.???? ?....

      4.???? ?/**

      5.???? ? *?如果在類中定義使用泛型的靜態(tài)方法,需要添加額外的泛型聲明(將這個(gè)方法定義成泛型方法)

      6.???? ? *?即使靜態(tài)方法要使用泛型類中已經(jīng)聲明過的泛型也不可以。

      7.???? ? *?如:public static void show(T t){..},此時(shí)編譯器會(huì)提示錯(cuò)誤信息:

      8.???? ? ? ? ?"StaticGenerator cannot be refrenced from static context"

      9.???? ? */

      10.???? ?publicstaticvoidshow(T t){

      11.

      12.???? ?}

      13.??}

      泛型方法能使方法獨(dú)立于類而產(chǎn)生變化,以下是一個(gè)基本的指導(dǎo)原則:

      無論何時(shí),如果你能做到,你就該盡量使用泛型方法。也就是說,如果使用泛型方法將整個(gè)類泛型化,那么就應(yīng)該使用泛型方法。另外對(duì)于一個(gè)static的方法而已,無法訪問泛型類型的參數(shù)。所以如果static方法要使用泛型能力,就必須使其成為泛型方法。

      泛型上下邊界

      在使用泛型的時(shí)候,我們還可以為傳入的泛型類型實(shí)參進(jìn)行上下邊界的限制,如:類型實(shí)參只準(zhǔn)傳入某種類型的父類或某種類型的子類。

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

      1.??publicvoidshowKeyValue1(Genericobj){

      2.???? ?Log.d("泛型測(cè)試","key value is "+obj.getKey());

      3.??}

      1.??Genericgeneric1?=newGeneric("11111");

      2.??Genericgeneric2?=newGeneric(2222);

      3.??Genericgeneric3?=newGeneric(2.4f);

      4.??Genericgeneric4?=newGeneric(2.56);

      5.

      6.??//這一行代碼編譯器會(huì)提示錯(cuò)誤,因?yàn)镾tring類型并不是Number類型的子類

      7.??//showKeyValue1(generic1);

      8.

      9.??showKeyValue1(generic2);

      10.??showKeyValue1(generic3);

      11.??showKeyValue1(generic4);

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

      1.??publicclassGeneric{

      2.???? ?privateT key;

      3.

      4.???? ?publicGeneric(T key){

      5.???? ? ? ?this.key?=key;

      0x6 Java系列:java 泛型詳解-絕對(duì)是對(duì)泛型方法講解最詳細(xì)的,沒有之一

      6.???? ?}

      7.

      8.???? ?publicT getKey(){

      9.???? ? ? ?returnkey;

      10.???? ?}

      11.??}

      //這一行代碼也會(huì)報(bào)錯(cuò),因?yàn)镾tring不是Number的子類

      1.??Genericgeneric1?=newGeneric("11111");

      再來一個(gè)泛型方法的例子:

      1.??//在泛型方法中添加上下邊界限制的時(shí)候,必須在權(quán)限聲明與返回值之間的上添加上下邊界,即在泛型聲明的時(shí)候添加

      2.??//public T showKeyName(Generic container),編譯器會(huì)報(bào)錯(cuò):"Unexpected bound"

      3.??publicT showKeyName(Genericcontainer){

      4.???? ?System.out.println("container key :"+container.getKey());

      5.???? ?T test?=container.getKey();

      6.???? ?returntest;

      7.??}

      通過上面的兩個(gè)例子可以看出:泛型的上下邊界添加,必須與泛型的聲明在一起。

      關(guān)于泛型數(shù)組要提一下

      看到了很多文章中都會(huì)提起泛型數(shù)組,經(jīng)過查看sun的說明文檔,在java中是”不能創(chuàng)建一個(gè)確切的泛型類型的數(shù)組”的。

      也就是說下面的這個(gè)例子是不可以的:

      1.??List[]ls?=newArrayList[10];

      而使用通配符創(chuàng)建泛型數(shù)組是可以的,如下面這個(gè)例子:

      1.??List[]ls?=newArrayList[10];

      這樣也是可以的:

      1.??List[]ls?=newArrayList[10];

      下面使用Sun的一篇文檔的一個(gè)例子來說明這個(gè)問題:

      1.??List[]lsa?=newList[10];// Not really allowed.

      2.??Objecto?=lsa;

      3.??Object[]oa?=(Object[])o;

      4.??Listli?=newArrayList();

      5.??li.add(newInteger(3));

      6.??oa[1]?=li;// Unsound, but passes run time store check

      7.??Strings?=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ò)。

      而對(duì)泛型數(shù)組的聲明進(jìn)行限制,對(duì)于這樣的情況,可以在編譯期提示代碼有類型安全問題,比沒有任何提示要強(qiáng)很多。

      下面采用通配符的方式是被允許的:數(shù)組的類型不可以是類型變量,除非是采用通配符的方式,因?yàn)閷?duì)于通配符的方式,最后取出數(shù)據(jù)是要做顯式的類型轉(zhuǎn)換的。

      1.??List[]lsa?=newList[10];// OK, array of unbounded wildcard type.

      2.??Objecto?=lsa;

      3.??Object[]oa?=(Object[])o;

      4.??Listli?=newArrayList();

      5.??li.add(newInteger(3));

      6.??oa[1]?=li;// Correct.

      7.??Integeri?=(Integer)lsa[1].get(0);// OK

      最后

      本文中的例子主要是為了闡述泛型中的一些思想而簡單舉出的,并不一定有著實(shí)際的可用性。另外,一提到泛型,相信大家用到最多的就是在集合中,其實(shí),在實(shí)際的編程過程中,自己可以使用泛型去簡化開發(fā),且能很好的保證代碼質(zhì)量。

      https://blog.csdn.net/s10461/article/details/53941091

      方志朋簡介:SpringCloud中國社區(qū)聯(lián)合創(chuàng)始人,博客訪問量突破一千萬,愛好開源,熱愛分享,活躍于各大社區(qū),保持著非常強(qiáng)的學(xué)習(xí)驅(qū)動(dòng)力,終身學(xué)習(xí)踐行者,終身學(xué)習(xí)受益者。目前就職于國內(nèi)某家知名互聯(lián)網(wǎng)保險(xiǎn)公司,擔(dān)任DEVOPS工程師,對(duì)微服務(wù)領(lǐng)域和持續(xù)集成領(lǐng)域研究較深,精通微服務(wù)框架SpringCloud

      版權(quán)聲明:本文內(nèi)容由網(wǎng)絡(luò)用戶投稿,版權(quán)歸原作者所有,本站不擁有其著作權(quán),亦不承擔(dān)相應(yīng)法律責(zé)任。如果您發(fā)現(xiàn)本站中有涉嫌抄襲或描述失實(shí)的內(nèi)容,請(qǐng)聯(lián)系我們jiasou666@gmail.com 處理,核實(shí)后本網(wǎng)站將在24小時(shí)內(nèi)刪除侵權(quán)內(nèi)容。

      版權(quán)聲明:本文內(nèi)容由網(wǎng)絡(luò)用戶投稿,版權(quán)歸原作者所有,本站不擁有其著作權(quán),亦不承擔(dān)相應(yīng)法律責(zé)任。如果您發(fā)現(xiàn)本站中有涉嫌抄襲或描述失實(shí)的內(nèi)容,請(qǐng)聯(lián)系我們jiasou666@gmail.com 處理,核實(shí)后本網(wǎng)站將在24小時(shí)內(nèi)刪除侵權(quán)內(nèi)容。

      上一篇:GtkApplication對(duì)象
      下一篇:藍(lán)橋杯——BASIC-11——十六進(jìn)制轉(zhuǎn)十進(jìn)制
      相關(guān)文章
      亚洲国产成人久久精品app| av在线亚洲欧洲日产一区二区| 亚洲免费一区二区| 99亚洲精品卡2卡三卡4卡2卡| 77777午夜亚洲| 亚洲一级毛片免费在线观看| 亚洲国产视频网站| 亚洲成人福利在线观看| 亚洲最大在线观看| 亚洲成aⅴ人在线观看| 亚洲国产精品久久久久秋霞影院| 亚洲精品美女视频| 亚洲一区在线视频| 亚洲综合中文字幕无线码| 亚洲综合丁香婷婷六月香| 亚洲高清有码中文字| 精品国产成人亚洲午夜福利| 亚洲性无码一区二区三区| 亚洲成AV人片在WWW| 亚洲国产精品碰碰| 亚洲综合精品网站| 国产成人A亚洲精V品无码| 亚洲欧洲美洲无码精品VA| 国产AV无码专区亚洲AVJULIA| 亚洲av永久无码精品表情包| 亚洲爱情岛论坛永久| 亚洲精品国产福利在线观看| 亚洲免费网站在线观看| 亚洲综合偷自成人网第页色| 亚洲av无码偷拍在线观看| jizzjizz亚洲| 亚洲精品成人片在线播放 | 亚洲春色在线视频| 亚洲日韩图片专区第1页| 亚洲人成在线播放| 亚洲精品无码永久在线观看男男| 国产亚洲视频在线观看| 中文字幕中韩乱码亚洲大片| 亚洲成A人片777777| 亚洲无线一二三四区| 亚洲一区AV无码少妇电影|