Java8中對(duì)泛型目標(biāo)類型推斷方法的改進(jìn)
一、簡(jiǎn)單理解泛型
泛型是Java SE 1.5的新特性,泛型的本質(zhì)是參數(shù)化類型,也就是說(shuō)所操作的數(shù)據(jù)類型被指定為一個(gè)參數(shù)。通俗點(diǎn)將就是“類型的變量”。這種類型變量可以用在類、接口和方法的創(chuàng)建中。
理解Java泛型最簡(jiǎn)單的方法是把它看成一種便捷語(yǔ)法,能節(jié)省你某些Java類型轉(zhuǎn)換(casting)上的操作:
上面的代碼自身已表達(dá)的很清楚:box是一個(gè)裝有Apple對(duì)象的List。get方法返回一個(gè)Apple對(duì)象實(shí)例,這個(gè)過(guò)程不需要進(jìn)行類型轉(zhuǎn)換。沒(méi)有泛型,上面的代碼需要寫(xiě)成這樣:
二、泛型的尷尬
泛型的最大優(yōu)點(diǎn)是提供了程序的類型安全同時(shí)可以向后兼容,但也有尷尬的地方,就是每次定義時(shí)都要寫(xiě)明泛型的類型,這樣顯示指定不僅感覺(jué)有些冗長(zhǎng),最主要是很多程序員不熟悉泛型,因此很多時(shí)候不能夠給出正確的類型參數(shù),現(xiàn)在通過(guò)編譯器自動(dòng)推斷泛型的參數(shù)類型,能夠減少這樣的情況,并提高代碼可讀性。
三、java7的泛型類型推斷改進(jìn)
在以前的版本中使用泛型類型,需要在聲明并賦值的時(shí)候,兩側(cè)都加上泛型類型。例如:
你可能覺(jué)得:老子在聲明變量的的時(shí)候已經(jīng)指明了參數(shù)類型,為毛還要在初始化對(duì)象時(shí)再指定?幸好,在Java SE 7中,這種方式得以改進(jìn),現(xiàn)在你可以使用如下語(yǔ)句進(jìn)行聲明并賦值:
在這條語(yǔ)句中,編譯器會(huì)根據(jù)變量聲明時(shí)的泛型類型自動(dòng)推斷出實(shí)例化HashMap時(shí)的泛型類型。再次提醒一定要注意new HashMap后面的“<>”,只有加上這個(gè)“<>”才表示是自動(dòng)類型推斷,否則就是非泛型類型的HashMap,并且在使用編譯器編譯源代碼時(shí)會(huì)給出一個(gè)警告提示。
但是:Java SE 7在創(chuàng)建泛型實(shí)例時(shí)的類型推斷是有限制的:只有構(gòu)造器的參數(shù)化類型在上下文中被顯著的聲明了,才可以使用類型推斷,否則不行。例如:下面的例子在java 7無(wú)法正確編譯(但現(xiàn)在在java8里面可以編譯,因?yàn)楦鶕?jù)方法參數(shù)來(lái)自動(dòng)推斷泛型的類型):
List<String> list = new ArrayList<>();
list.add("A");// 由于addAll期望獲得Collection<? extends String>類型的參數(shù),因此下面的語(yǔ)句無(wú)法通過(guò)
list.addAll(new ArrayList<>());
四、Java8的泛型類型推斷改進(jìn)
java8里面泛型的目標(biāo)類型推斷主要2個(gè):
1.支持通過(guò)方法上下文推斷泛型目標(biāo)類型
2.支持在方法調(diào)用鏈路當(dāng)中,泛型類型推斷傳遞到最后一個(gè)方法
讓我們看看官網(wǎng)的例子:
class List<E> {
static <Z> List<Z> nil() { ... };
static <Z> List<Z> cons(Z head, List<Z> tail) { ... };
E head() { ... }
}
根據(jù)JEP101的特性,我們?cè)谡{(diào)用上面方法的時(shí)候可以這樣寫(xiě)
//通過(guò)方法賦值的目標(biāo)參數(shù)來(lái)自動(dòng)推斷泛型的類型
List<String> l = List.nil();
//而不是顯示的指定類型
//List<String> l = List.<String>nil();
//通過(guò)前面方法參數(shù)類型推斷泛型的類型
List.cons(42, List.nil());
//而不是顯示的指定類型
//List.cons(42, List.<Integer>nil());
五、總結(jié)
以上是JEP101的特性內(nèi)容了,Java作為靜態(tài)語(yǔ)言的代表者,可以說(shuō)類型系統(tǒng)相當(dāng)豐富。導(dǎo)致類型間互相轉(zhuǎn)換的問(wèn)題困擾著每個(gè)java程序員,通過(guò)編譯器自動(dòng)推斷類型的東西可以稍微緩解一下類型轉(zhuǎn)換太復(fù)雜的問(wèn)題。 雖然說(shuō)是小進(jìn)步,但對(duì)于我們天天寫(xiě)代碼的程序員,肯定能帶來(lái)巨大的作用,至少心情更愉悅了~~說(shuō)不定在java 9里面,我們會(huì)得到一個(gè)通用的類型var,像js或者scala的一些動(dòng)態(tài)語(yǔ)言那樣^_^
相關(guān)文章
Java 動(dòng)態(tài)編譯在項(xiàng)目中的實(shí)踐分享
在 Java 中,動(dòng)態(tài)編譯是指在運(yùn)行時(shí)動(dòng)態(tài)地編譯 Java 源代碼,生成字節(jié)碼,并加載到 JVM 中執(zhí)行,動(dòng)態(tài)編譯可以用于實(shí)現(xiàn)動(dòng)態(tài)代碼生成、動(dòng)態(tài)加載、插件化等功能,本文將給大家分享一下Java 動(dòng)態(tài)編譯在項(xiàng)目中的實(shí)踐,感興趣的同學(xué)跟著小編一起來(lái)看看吧2023-07-07Java OpenCV實(shí)現(xiàn)人臉識(shí)別過(guò)程詳解
這篇文章主要介紹了Java OpenCV實(shí)現(xiàn)人臉識(shí)別過(guò)程詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-08-08Java多線程 兩階段終止模式Two-Phase Termination Patter
這篇文章主要介紹了Java多線程 兩階段終止模式Two-Phase Termination Patter,該模式有兩個(gè)角色,分別是Terminator,終止者,負(fù)責(zé)接收終止請(qǐng)求,執(zhí)行終止處理,處理完成后再終止自己。TerminationRequester終止請(qǐng)求發(fā)出者,用來(lái)向Terminator發(fā)出終止請(qǐng)求,需要的朋友可以參考一下2021-10-10Mybatis通過(guò)Mapper代理連接數(shù)據(jù)庫(kù)的方法
這篇文章主要介紹了Mybatis通過(guò)Mapper代理連接數(shù)據(jù)庫(kù)的方法,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-11-11Mybatis-Plus雪花id的使用以及解析機(jī)器ID和數(shù)據(jù)標(biāo)識(shí)ID實(shí)現(xiàn)
這篇文章主要介紹了Mybatis-Plus雪花id的使用以及解析機(jī)器ID和數(shù)據(jù)標(biāo)識(shí)ID實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08處理@PathVariable注解允許參數(shù)為空、允許不傳參數(shù)的問(wèn)題
這篇文章主要介紹了處理@PathVariable注解允許參數(shù)為空、允許不傳參數(shù)的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-02-02Java實(shí)現(xiàn)過(guò)濾掉map集合中key或value為空的值示例
這篇文章主要介紹了Java實(shí)現(xiàn)過(guò)濾掉map集合中key或value為空的值,涉及java針對(duì)map的簡(jiǎn)單遍歷、判斷、移除等相關(guān)操作技巧,需要的朋友可以參考下2018-06-06Spring學(xué)習(xí)之開(kāi)發(fā)環(huán)境搭建的詳細(xì)步驟
本篇文章主要介紹了Spring學(xué)習(xí)之開(kāi)發(fā)環(huán)境搭建的詳細(xì)步驟,具有一定的參考價(jià)值,有興趣的可以了解一下2017-07-07