久久国产成人av_抖音国产毛片_a片网站免费观看_A片无码播放手机在线观看,色五月在线观看,亚洲精品m在线观看,女人自慰的免费网址,悠悠在线观看精品视频,一级日本片免费的,亚洲精品久,国产精品成人久久久久久久

分享

揭開Java 泛型類型擦除神秘面紗

 goldbomb 2018-09-18

泛型,,一個(gè)孤獨(dú)的守門者,。

大家可能會(huì)有疑問(wèn),我為什么叫做泛型是一個(gè)守門者,。這其實(shí)是我個(gè)人的看法而已,我的意思是說(shuō)泛型沒(méi)有其看起來(lái)那么深不可測(cè),,它并不神秘與神奇。泛型是 Java 中一個(gè)很小巧的概念,,但同時(shí)也是一個(gè)很容易讓人迷惑的知識(shí)點(diǎn),它讓人迷惑的地方在于它的許多表現(xiàn)有點(diǎn)違反直覺(jué),。

文章開始的地方,,先給大家奉上一道經(jīng)典的測(cè)試題。

List l1 = new ArrayList();
List l2 = new ArrayList();

System.out.println(l1.getClass() == l2.getClass());


請(qǐng)問(wèn),上面代碼最終結(jié)果輸出的是什么,?不了解泛型的和很熟悉泛型的同學(xué)應(yīng)該能夠答出來(lái),,而對(duì)泛型有所了解,,但是了解不深入的同學(xué)可能會(huì)答錯(cuò)。

正確答案是 true,。

上面的代碼中涉及到了泛型,而輸出的結(jié)果緣由是類型擦除,。先好好說(shuō)說(shuō)泛型,。

泛型是什么?

泛型的英文是 generics,,generic 的意思是通用,而翻譯成中文,泛應(yīng)該意為廣泛,,型是類型,。所以泛型就是能廣泛適用的類型,。

但泛型還有一種較為準(zhǔn)確的說(shuō)法就是為了參數(shù)化類型,,或者說(shuō)可以將類型當(dāng)作參數(shù)傳遞給一個(gè)類或者是方法,。

那么,,如何解釋類型參數(shù)化呢,?

public class Cache {
   Object value;

   public Object getValue({
       return value;
   }

   public void setValue(Object value{
       this.value = value;
   }

}


假設(shè) Cache 能夠存取任何類型的值,于是,,我們可以這樣使用它,。

Cache cache = new Cache();
cache.setValue(134);
int value = (int) cache.getValue();
cache.setValue('hello');
String value1 = (String) cache.getValue();


使用的方法也很簡(jiǎn)單,,只要我們做正確的強(qiáng)制轉(zhuǎn)換就好了,。

但是,,泛型卻給我們帶來(lái)了不一樣的編程體驗(yàn)。

public class Cache {
   T value;

   public Object getValue() {
       return value;
   }

   public void setValue(T value) {
       this.value = value;
   }

}


這就是泛型,它將 value 這個(gè)屬性的類型也參數(shù)化了,,這就是所謂的參數(shù)化類型。再看它的使用方法,。

CacheString> cache1 = new CacheString>();
cache1.setValue('123');
String value2 = cache1.getValue();

CacheInteger> cache2 = new CacheInteger>();
cache2.setValue(456);
int value3 = cache2.getValue();


最顯而易見的好處就是它不再需要對(duì)取出來(lái)的結(jié)果進(jìn)行強(qiáng)制轉(zhuǎn)換了,。但,,還有另外一點(diǎn)不同,。 



泛型除了可以將類型參數(shù)化外,而參數(shù)一旦確定好,,如果類似不匹配,,編譯器就不通過(guò),。 
上面代碼顯示,無(wú)法將一個(gè) String 對(duì)象設(shè)置到 cache2 中,,因?yàn)榉盒妥屗唤邮?Integer 的類型,。

所以,綜合上面信息,,我們可以得到下面的結(jié)論,。

  1. 與普通的 Object 代替一切類型這樣簡(jiǎn)單粗暴而言,,泛型使得數(shù)據(jù)的類別可以像參數(shù)一樣由外部傳遞進(jìn)來(lái),。它提供了一種擴(kuò)展能力。它更符合面向抽象開發(fā)的軟件編程宗旨,。

  2. 當(dāng)具體的類型確定后,,泛型又提供了一種類型檢測(cè)的機(jī)制,只有相匹配的數(shù)據(jù)才能正常的賦值,,否則編譯器就不通過(guò)。所以說(shuō),,它是一種類型安全檢測(cè)機(jī)制,一定程度上提高了軟件的安全性防止出現(xiàn)低級(jí)的失誤,。

  3. 泛型提高了程序代碼的可讀性,,不必要等到運(yùn)行的時(shí)候才去強(qiáng)制轉(zhuǎn)換,在定義或者實(shí)例化階段,,因?yàn)?nbsp;Cache 這個(gè)類型顯化的效果,,程序員能夠一目了然猜測(cè)出代碼要操作的數(shù)據(jù)類型,。


下面的文章,我們正常介紹泛型的相關(guān)知識(shí),。

泛型的定義和使用

泛型按照使用情況可以分為 3 種。 
1. 泛型類,。 
2. 泛型方法。 
3. 泛型接口,。

泛型類

我們可以這樣定義一個(gè)泛型類,。

public class Test {
   T field1;
}


尖括號(hào) <> 中的 T 被稱作是類型參數(shù),用于指代任何類型,。事實(shí)上,,T 只是一種習(xí)慣性寫法,如果你愿意,。你可以這樣寫,。

public class Test {
   Hello field1;
}


但出于規(guī)范的目的,,Java 還是建議我們用單個(gè)大寫字母來(lái)代表類型參數(shù)。常見的如: 
1. T 代表一般的任何類,。 
2. E 代表 Element 的意思,,或者 Exception 異常的意思。 
3. K 代表 Key 的意思,。 
4. V 代表 Value 的意思,通常與 K 一起配合使用,。 
5. S 代表 Subtype 的意思,文章后面部分會(huì)講解示意,。

如果一個(gè)類被  的形式定義,,那么它就被稱為是泛型類,。

那么對(duì)于泛型類怎么樣使用呢,?

TestString> test1 = new Test<>();
TestInteger> test2 = new Test<>();


只要在對(duì)泛型類創(chuàng)建實(shí)例的時(shí)候,在尖括號(hào)中賦值相應(yīng)的類型便是,。T 就會(huì)被替換成對(duì)應(yīng)的類型,,如 String 或者是 Integer。你可以相像一下,當(dāng)一個(gè)泛型類被創(chuàng)建時(shí),,內(nèi)部自動(dòng)擴(kuò)展成下面的代碼,。


public class Test {
   String field1;
}


當(dāng)然,泛型類不至接受一個(gè)類型參數(shù),,它還可以這樣接受多個(gè)類型參數(shù),。

public class MultiType {
   E value1;
   T value2;

   public E getValue1(){
       return value1;
   }

   public T getValue2(){
       return value2;
   }
}


泛型方法

public class Test1 {

   public  void testMethod(T t){

   }
}


泛型方法與泛型類稍有不同的地方是,類型參數(shù)也就是尖括號(hào)那一部分是寫在返回值前面的,。 中的 T 被稱為類型參數(shù),,而方法中的 T 被稱為參數(shù)化類型,它不是運(yùn)行時(shí)真正的參數(shù),。

當(dāng)然,,聲明的類型參數(shù),其實(shí)也是可以當(dāng)作返回值的類型的,。

public   testMethod1(T t){
       return null;
}


泛型類與泛型方法的共存現(xiàn)象

public class Test1{

   public  void testMethod(T t){
       System.out.println(t.getClass().getName());
   }
   public   testMethod1(T t){
       return t;
   }
}


上面代碼中,,Test1 是泛型類,testMethod 是泛型類中的普通方法,,而 testMethod1 是一個(gè)泛型方法,。而泛型類中的類型參數(shù)與泛型方法中的類型參數(shù)是沒(méi)有相應(yīng)的聯(lián)系的,泛型方法始終以自己定義的類型參數(shù)為準(zhǔn),。

所以,,針對(duì)上面的代碼,我們可以這樣編寫測(cè)試代碼,。

Test1String> t = new Test1();
t.testMethod('generic');
Integer i = t.testMethod1(new Integer(1));


泛型類的實(shí)際類型參數(shù)是 String,,而傳遞給泛型方法的類型參數(shù)是 Integer,兩者不想干,。

但是,,為了避免混淆,如果在一個(gè)泛型類中存在泛型方法,,那么兩者的類型參數(shù)最好不要同名,。比如,Test1 代碼可以更改為這樣

public class Test1{

   public  void testMethod(T t){
       System.out.println(t.getClass().getName());
   }
   public   testMethod1(E e){
       return e;
   }
}


泛型接口

泛型接口和泛型類差不多,,所以一筆帶過(guò),。

public interface IterableT{
}


通配符 ?

除了用  表示泛型外,,還有  這種形式,。 被稱為通配符,。

可能有同學(xué)會(huì)想,,已經(jīng)有了  的形式了,,為什么還要引進(jìn)  這樣的概念呢?

class Base{}

class Sub extends Base{}

Sub sub = new Sub();
Base base = sub;


上面代碼顯示,,Base 是 Sub 的父類,,它們之間是繼承關(guān)系,所以 Sub 的實(shí)例可以給一個(gè) Base 引用賦值,,那么

List lsub = new ArrayList<>();
List lbase = lsub;


最后一行代碼成立嗎,?編譯會(huì)通過(guò)嗎?

答案是否定的,。

編譯器不會(huì)讓它通過(guò)的,。Sub 是 Base 的子類,不代表 List 和 List 有繼承關(guān)系,。

但是,,在現(xiàn)實(shí)編碼中,確實(shí)有這樣的需求,,希望泛型能夠處理某一范圍內(nèi)的數(shù)據(jù)類型,,比如某個(gè)類和它的子類,對(duì)此 Java 引入了通配符這個(gè)概念,。

所以,,通配符的出現(xiàn)是為了指定泛型中的類型范圍

通配符有 3 種形式,。

  1.  被稱作無(wú)限定的通配符,。

  2.  被稱作有上限的通配符。

  3.  被稱作有下限的通配符,。


無(wú)限定通配符

public void testWildCards(Collection collection){
}


上面的代碼中,,方法內(nèi)的參數(shù)是被無(wú)限定通配符修飾的 Collection 對(duì)象,它隱略地表達(dá)了一個(gè)意圖或者可以說(shuō)是限定,,那就是 testWidlCards() 這個(gè)方法內(nèi)部無(wú)需關(guān)注 Collection 中的真實(shí)類型,,因?yàn)樗俏粗?/span>。所以,,你只能調(diào)用 Collection 中與類型無(wú)關(guān)的方法,。

我們可以看到,當(dāng)  存在時(shí),,Collection 對(duì)象喪失了 add() 方法的功能,編譯器不通過(guò),。 
我們?cè)倏创a,。

List<>> wildlist = new ArrayList();
wildlist.add(123);// 編譯不通過(guò)


有人說(shuō), 提供了只讀的功能,,也就是它刪減了增加具體類型元素的能力,,只保留與具體類型無(wú)關(guān)的功能,。它不管裝載在這個(gè)容器內(nèi)的元素是什么類型,它只關(guān)心元素的數(shù)量,、容器是否為空,?我想這種需求還是很常見的吧。

有同學(xué)可能會(huì)想,, 既然作用這么渺小,,那么為什么還要引用它呢?

個(gè)人認(rèn)為,,提高了代碼的可讀性,,程序員看到這段代碼時(shí),就能夠迅速對(duì)此建立極簡(jiǎn)潔的印象,,能夠快速推斷源碼作者的意圖,。

 代表著類型未知,但是我們的確需要對(duì)于類型的描述再精確一點(diǎn),,我們希望在一個(gè)范圍內(nèi)確定類別,,比如類型 A 及 類型 A 的子類都可以。

public void testSub(Collection para){

}


上面代碼中,,para 這個(gè) Collection 接受 Base 及 Base 的子類的類型,。

但是,它仍然喪失了寫操作的能力,。也就是說(shuō)

para.add(new Sub());
para.add(new Base());


仍然編譯不通過(guò),。

沒(méi)有關(guān)系,我們不知道具體類型,,但是我們至少清楚了類型的范圍,。

這個(gè)和  相對(duì)應(yīng),代表 T 及 T 的超類,。

public void testSuper(Collection<>super Sub> para){
}


 神奇的地方在于,,它擁有一定程度的寫操作的能力。

public void testSuper(Collection para){
   para.add(new Sub());//編譯通過(guò)
   para.add(new Base());//編譯不通過(guò)
}


通配符與類型參數(shù)的區(qū)別

一般而言,,通配符能干的事情都可以用類型參數(shù)替換,。 
比如

public void testWildCards(Collection collection){}


可以被

public  void test(Collection collection){}


取代。

值得注意的是,,如果用泛型方法來(lái)取代通配符,,那么上面代碼中 collection 是能夠進(jìn)行寫操作的。只不過(guò)要進(jìn)行強(qiáng)制轉(zhuǎn)換,。

public  void test(Collection collection){
   collection.add((T)new Integer(12));
   collection.add((T)'123');
}


需要特別注意的是,,類型參數(shù)適用于參數(shù)之間的類別依賴關(guān)系,舉例說(shuō)明,。

public class Test2 T,E extends T>{
   T value1;
   E value2;
}


public  void test(D d,S s){

   }


E 類型是 T 類型的子類,,顯然這種情況類型參數(shù)更適合,。 
有一種情況是,通配符和類型參數(shù)一起使用,。

public  void test(T t,Collection collection){

}


如果一個(gè)方法的返回類型依賴于參數(shù)的類型,,那么通配符也無(wú)能為力。

public T test1(T t){
   return value1;
}


類型擦除

泛型是 Java 1.5 版本才引進(jìn)的概念,,在這之前是沒(méi)有泛型的概念的,,但顯然,泛型代碼能夠很好地和之前版本的代碼很好地兼容,。

這是因?yàn)椋?span>泛型信息只存在于代碼編譯階段,,在進(jìn)入 JVM 之前,與泛型相關(guān)的信息會(huì)被擦除掉,,專業(yè)術(shù)語(yǔ)叫做類型擦除,。

通俗地講,泛型類和普通類在 java 虛擬機(jī)內(nèi)是沒(méi)有什么特別的地方,?;仡櫸恼麻_始時(shí)的那段代碼

List l1 = new ArrayList();
List l2 = new ArrayList();

System.out.println(l1.getClass() == l2.getClass());


打印的結(jié)果為 true 是因?yàn)?nbsp;List 和 List 在 jvm 中的 Class 都是 List.class。

泛型信息被擦除了,。

可能同學(xué)會(huì)問(wèn),,那么類型 String 和 Integer 怎么辦?

答案是泛型轉(zhuǎn)譯,。

public class Erasure {
   T object;

   public Erasure(T object) {
       this.object = object;
   }

}


Erasure 是一個(gè)泛型類,,我們查看它在運(yùn)行時(shí)的狀態(tài)信息可以通過(guò)反射。

ErasureString> erasure = new ErasureString>('hello');
Class eclz = erasure.getClass();
System.out.println('erasure class is:'+eclz.getName());


打印的結(jié)果是

erasure class is:com.frank.test.Erasure


Class 的類型仍然是 Erasure 并不是 Erasure 這種形式,,那我們?cè)倏纯捶盒皖愔?T 的類型在 jvm 中是什么具體類型,。

Field[] fs = eclz.getDeclaredFields();
for ( Field f:fs) {
   System.out.println('Field name '+f.getName()+' type:'+f.getType().getName());
}


打印結(jié)果是

Field name object type:java.lang.Object


那我們可不可以說(shuō),泛型類被類型擦除后,,相應(yīng)的類型就被替換成 Object 類型呢,?

這種說(shuō)法,不完全正確,。

我們更改一下代碼,。

public class Erasure T extends String>{
//  public class Erasure {
   T object;

   public Erasure(T object) {
       this.object = object;
   }

}


現(xiàn)在再看測(cè)試結(jié)果:

Field name object type:java.lang.String


我們現(xiàn)在可以下結(jié)論了,在泛型類被類型擦除的時(shí)候,,之前泛型類中的類型參數(shù)部分如果沒(méi)有指定上限,,如  則會(huì)被轉(zhuǎn)譯成普通的 Object 類型,如果指定了上限如  則類型參數(shù)就被替換成類型上限,。

所以,,在反射中。

public class Erasure {
   T object;

   public Erasure(T object) {
       this.object = object;
   }

   public void add(T object){

   }

}


add() 這個(gè)方法對(duì)應(yīng)的 Method 的簽名應(yīng)該是 Object.class,。

Erasure erasure = new Erasure('hello');
Class eclz = erasure.getClass();
System.out.println('erasure class is:'+eclz.getName());

Method[] methods = eclz.getDeclaredMethods();
for ( Method m:methods ){
   System.out.println(' method:'+m.toString());
}


打印結(jié)果是

method:public void com.frank.test.Erasure.add(java.lang.Object)


也就是說(shuō),,如果你要在反射中找到 add 對(duì)應(yīng)的 Method,你應(yīng)該調(diào)用 getDeclaredMethod('add',Object.class) 否則程序會(huì)報(bào)錯(cuò),,提示沒(méi)有這么一個(gè)方法,,原因就是類型擦除的時(shí)候,T 被替換成 Object 類型了,。

類型擦除帶來(lái)的局限性

類型擦除,,是泛型能夠與之前的 java 版本代碼兼容共存的原因。但也因?yàn)轭愋筒脸?,它?huì)抹掉很多繼承相關(guān)的特性,,這是它帶來(lái)的局限性。

理解類型擦除有利于我們繞過(guò)開發(fā)當(dāng)中可能遇到的雷區(qū),,同樣理解類型擦除也能讓我們繞過(guò)泛型本身的一些限制,。比如

正常情況下,因?yàn)榉盒偷南拗?,編譯器不讓最后一行代碼編譯通過(guò),,因?yàn)轭愃撇黄ヅ洌?,基于?duì)類型擦除的了解,,利用反射,我們可以繞過(guò)這個(gè)限制,。

public interface ListEextends CollectionE>{

    boolean add(E e);
}


上面是 List 和其中的 add() 方法的源碼定義,。

因?yàn)?E 代表任意的類型,所以類型擦除時(shí),,add 方法其實(shí)等同于

boolean add(Object obj);


那么,,利用反射,我們繞過(guò)編譯器去調(diào)用 add 方法,。

public class ToolTest {


   public static void main(String[] args{
       List ls = new ArrayList<>();
       ls.add(23);
//      ls.add('text');
       try {
           Method method = ls.getClass().getDeclaredMethod('add',Object.class);


           method.invoke(ls,'test');
           method.invoke(ls,42.9f);
       } catch (NoSuchMethodException e) {
           // TODO Auto-generated catch block
           e.printStackTrace();
       } catch (SecurityException e) {
           // TODO Auto-generated catch block
           e.printStackTrace();
       } catch (IllegalAccessException e) {
           // TODO Auto-generated catch block
           e.printStackTrace();
       } catch (IllegalArgumentException e) {
           // TODO Auto-generated catch block
           e.printStackTrace();
       } catch (InvocationTargetException e) {
           // TODO Auto-generated catch block
           e.printStackTrace();
       }

       for ( Object o: ls){
           System.out.println(o);
       }

   }

}


打印結(jié)果是:

23
test
42.9


可以看到,,利用類型擦除的原理,用反射的手段就繞過(guò)了正常開發(fā)中編譯器不允許的操作限制,。

泛型中值得注意的地方

泛型類或者泛型方法中,,不接受 8 種基本數(shù)據(jù)類型。

所以,,你沒(méi)有辦法進(jìn)行這樣的編碼,。

List li = new ArrayList<>();
List li = new ArrayList<>();


需要使用它們對(duì)應(yīng)的包裝類。

List li = new ArrayList<>();
List li1 = new ArrayList<>();


對(duì)泛型方法的困惑

public  test(T t){
   return null;
}


有的同學(xué)可能對(duì)于連續(xù)的兩個(gè) T 感到困惑,,其實(shí)  是為了說(shuō)明類型參數(shù),,是聲明,而后面的不帶尖括號(hào)的 T 是方法的返回值類型。 
你可以相像一下,,如果 test() 這樣被調(diào)用

test('123');


那么實(shí)際上相當(dāng)于

public String test(String t);


Java 不能創(chuàng)建具體類型的泛型數(shù)組

這句話可能難以理解,,代碼說(shuō)明,。

List[] li2 = new ArrayList[];
List li3 = new ArrayList[];


這兩行代碼是無(wú)法在編譯器中編譯通過(guò)的。原因還是類型擦除帶來(lái)的影響,。

List 和 List 在 jvm 中等同于List ,,所有的類型信息都被擦除,程序也無(wú)法分辨一個(gè)數(shù)組中的元素類型具體是 List類型還是 List 類型,。

但是,,

List<>>[] li3 = new ArrayList<>>[10];
li3[1] = new ArrayList();
List<>> v = li3[1];


借助于無(wú)限定通配符卻可以,前面講過(guò) ,? 代表未知類型,,所以它涉及的操作都基本上與類型無(wú)關(guān),因此 jvm 不需要針對(duì)它對(duì)類型作判斷,,因此它能編譯通過(guò),,但是,只提供了數(shù)組中的元素因?yàn)橥ㄅ浞?,它只能讀,,不能寫。比如,,上面的 v 這個(gè)局部變量,,它只能進(jìn)行 get() 操作,不能進(jìn)行 add() 操作,,這個(gè)在前面通配符的內(nèi)容小節(jié)中已經(jīng)講過(guò),。

泛型,并不神奇

我們可以看到,,泛型其實(shí)并沒(méi)有什么神奇的地方,,泛型代碼能做的非泛型代碼也能做。

而類型擦除,,是泛型能夠與之前的 java 版本代碼兼容共存的原因,。

可量也正因?yàn)轭愋筒脸龑?dǎo)致了一些隱患與局限。

但,,我還是要建議大家使用泛型,,如官方文檔所說(shuō)的,如果可以使用泛型的地方,,盡量使用泛型,。

畢竟它抽離了數(shù)據(jù)類型與代碼邏輯,本意是提高程序代碼的簡(jiǎn)潔性和可讀性,,并提供可能的編譯時(shí)類型轉(zhuǎn)換安全檢測(cè)功能,。

類型擦除不是泛型的全部,但是它卻能很好地檢測(cè)我們對(duì)于泛型這個(gè)概念的理解程度。

我在文章開頭將泛型比作是一個(gè)守門人,,原因就是他本意是好的,,守護(hù)我們的代碼安全,然后在門牌上寫著出入的各項(xiàng)規(guī)定,,及“xxx 禁止出入”的提醒,。但是同我們?nèi)粘K龅降哪切╅T衛(wèi)一般,他們古怪偏執(zhí),,死板守舊,我們可以利用反射基于類型擦除的認(rèn)識(shí),,來(lái)繞過(guò)泛型中某些限制,,現(xiàn)實(shí)生活中,也總會(huì)有調(diào)皮搗蛋者能夠基于對(duì)門衛(wèi)們生活作息的規(guī)律,,選擇性地繞開他們的監(jiān)視,,另辟蹊徑溜進(jìn)或者溜出大門,然后揚(yáng)長(zhǎng)而去,,剩下守衛(wèi)者一個(gè)孤獨(dú)的身影,。

所以,我說(shuō)泛型,,并不神秘,,也不神奇

    本站是提供個(gè)人知識(shí)管理的網(wǎng)絡(luò)存儲(chǔ)空間,,所有內(nèi)容均由用戶發(fā)布,,不代表本站觀點(diǎn)。請(qǐng)注意甄別內(nèi)容中的聯(lián)系方式,、誘導(dǎo)購(gòu)買等信息,,謹(jǐn)防詐騙。如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,,請(qǐng)點(diǎn)擊一鍵舉報(bào),。
    轉(zhuǎn)藏 分享 獻(xiàn)花(0

    0條評(píng)論

    發(fā)表

    請(qǐng)遵守用戶 評(píng)論公約

    類似文章 更多