一、泛型擦除
泛型擦除介紹
Java 的泛型是偽泛型,這是因?yàn)?Java 在運(yùn)行期間,所有的泛型信息都會(huì)被擦掉,這也就是通常所說(shuō)類型擦除。Java 泛型(generics) 是 JDK 5 中引入的一個(gè)新特性, 泛型提供了編譯時(shí)類型安全檢測(cè)機(jī)制。該機(jī)制允許程序員在編譯時(shí)檢測(cè)到非法的類型。泛型的本質(zhì)是參數(shù)化類型,也就是說(shuō)所操作的數(shù)據(jù)類型被指定為一個(gè)參數(shù)。
List
list.add(12);
//這里直接添加會(huì)報(bào)錯(cuò)
list.add(“a”);
Class extends List> clazz = list.getClass();
Method add = clazz.getDeclaredMethod(“add”, Object.class);
//但是通過(guò)反射添加是可以的
//這就說(shuō)明在運(yùn)行期間所有的泛型信息都會(huì)被擦掉
add.invoke(list, “kl”);
System.out.println(list);
帶來(lái)什么樣的問(wèn)題
(1) 強(qiáng)制類型轉(zhuǎn)化
這個(gè)問(wèn)題的結(jié)果我們已經(jīng)在上述文章中提及到了,通過(guò)反射的方式去進(jìn)行插入的時(shí)候,我們的數(shù)據(jù)就會(huì)發(fā)生錯(cuò)誤。
如果我們?cè)谝粋€(gè)List
(2)引用傳遞問(wèn)題
上面的問(wèn)題中,我們已經(jīng)說(shuō)過(guò)了T將在后期被轉(zhuǎn)義成Object,那我們對(duì)引用也進(jìn)行一個(gè)轉(zhuǎn)化,是否行得通呢?
List
List
如果你這樣寫(xiě),在我們的檢查階段,會(huì)報(bào)錯(cuò)。但是從邏輯意義上來(lái)說(shuō),其實(shí)你真的有錯(cuò)嗎?
假設(shè)說(shuō)我們的名列前茅種方案是正確的,那么其實(shí)就是將一堆Object數(shù)據(jù)存入,然后再由上面所說(shuō)的強(qiáng)制轉(zhuǎn)化一般,轉(zhuǎn)化成String類型,聽(tīng)起來(lái)完全ok,因?yàn)樵贚ist中本來(lái)存儲(chǔ)數(shù)據(jù)的方式就是Object。但其實(shí)是會(huì)出現(xiàn)ClassCastException的問(wèn)題,因?yàn)镺bject是萬(wàn)物的基類,但是強(qiáng)轉(zhuǎn)是為子類向父類準(zhǔn)備的措施。
再來(lái)假設(shè)說(shuō)我們的第二種方案是正確的,這個(gè)時(shí)候,根據(jù)上方的數(shù)據(jù)String存入,但是有什么意義存在呢?最后都還是要成Object的,你還不如就直接是Object。
延伸閱讀:
二、繼承型的用處是什么
其實(shí)他期待的就是這整個(gè)列表的數(shù)據(jù)的基礎(chǔ)都是來(lái)自我們的Parent,這樣獲取的數(shù)據(jù)全部人的父類其實(shí)都是來(lái)自于我們的Parent了,你可以叫這個(gè)列表為Parent家族。所以也可以說(shuō)這是一個(gè)適合頻繁讀取的方案。
Plate extends Fruit> p1=new Plate
Plate extends Fruit> p2=new Plate
// 修改數(shù)據(jù)不通過(guò)
p1.set(new Banana());
// 數(shù)據(jù)獲取一切正常
// 但是他只能精確到由我們定義的Fruit
Fruit result = p1.get();