java8到j(luò)ava15的新功能簡(jiǎn)介
本博文將為您提供自Java 7以來(lái)增加的很棒的新功能的示例。我將展示每個(gè)Java版本的至少一項(xiàng)重大改進(jìn),一直到2020年秋季發(fā)布的Java 15都有。Java現(xiàn)在完全支持lambda和函數(shù)式編程,類型推斷通過(guò)var,具有簡(jiǎn)單構(gòu)造函數(shù)的不可變集合以及多行字符串。此外,還有令人興奮的實(shí)驗(yàn)新功能,例如數(shù)據(jù)類(record)和sealed類。最后,我將討論Java REPL,它為快速實(shí)驗(yàn)提供了很高的價(jià)值。
函數(shù)式編程(Java 8)
在Java 8中,功能編程和lambda被添加為語(yǔ)言功能。函數(shù)式編程的兩個(gè)核心范例是不變值和將函數(shù)提升為一等公民的方法。數(shù)據(jù)經(jīng)過(guò)一系列修改步驟,其中每個(gè)步驟都需要一些輸入并將其映射到新的輸出。函數(shù)式編程可與Java中的Streamsnull安全monads(Optional)一起使用,如下所示...
流(Java 8)
對(duì)于一般的計(jì)算機(jī)程序,通常必須使用值列表,并對(duì)每個(gè)值執(zhí)行給定的轉(zhuǎn)換。在Java 8之前,您必須使用for循環(huán)進(jìn)行此轉(zhuǎn)換,但是從現(xiàn)在開(kāi)始,您可以使用Streams以下方法:
Stream.of("hello", "great") .map(s -> s + " world") .forEach(System.out::println); > hello world > great world
該map函數(shù)以一個(gè)lambda作為輸入,它將應(yīng)用于流中的所有元素。
Streams可以在Lists,Sets和Maps(通過(guò)轉(zhuǎn)換)上工作。多虧了Streams,您可以擺脫代碼中幾乎所有的循環(huán)!
Optional(Java 8)
Java中的另一個(gè)常見(jiàn)問(wèn)題是Null Pointer Exceptions。因此,Java引入了Optional –這是一個(gè)monad,它包裝了一個(gè)可能為null或不為null的引用??梢酝ㄟ^(guò)函數(shù)性方式將更新應(yīng)用于此Optional:
Optional.of(new Random().nextInt(10)) .filter(i -> i % 2 == 0) .map(i -> "number is even: " + i) .ifPresent(System.out::println); > number is even: 6
在上面的代碼段中,我們創(chuàng)建一個(gè)隨機(jī)數(shù),將其包裝在Optional對(duì)象中,然后僅打印偶數(shù)。
JShell(Java 9)
最后,我們有一個(gè)Java的REPL,它的名字叫JShell!簡(jiǎn)而言之,JShell允許在不編寫(xiě)和編譯完整Java類的情況下嘗試Java代碼段。相反,您可以一次執(zhí)行一個(gè)命令,然后立即看到結(jié)果。這是一個(gè)簡(jiǎn)單的例子:
$ <JDK>/bin/jshell jshell> System.out.println("hello world") hello world
長(zhǎng)期以來(lái),熟悉JavaScript或Python等解釋型語(yǔ)言的人們都對(duì)REPL感到滿意,但到目前為止,Java中缺少此功能。JShell允許定義變量,但也可以定義更復(fù)雜的實(shí)體,例如多行函數(shù),類和執(zhí)行循環(huán)。而且,JShell支持自動(dòng)完成,如果您不知道給定Java類提供的確切方法,該功能將非常有用。
不可變集合的工廠方法(Java 9)
ListsJava的簡(jiǎn)單初始化早已丟失,但現(xiàn)在已經(jīng)過(guò)去了。以前,您必須執(zhí)行以下操作:
jshell> List<Integer> list = Arrays.asList(1, 2, 3, 4) list ==> [1, 2, 3, 4]
現(xiàn)在將其簡(jiǎn)化如下:
jshell> List<Integer> list = List.of(1, 2, 3, 4) b ==> [1, 2, 3, 4]
List,Set和Mapof(...)存在這種奇特的方法。它們都只用一行簡(jiǎn)單的代碼就創(chuàng)建了一個(gè)不變的對(duì)象。
使用var(Java 10)進(jìn)行 類型推斷
Java 10引入了新的var關(guān)鍵字,該關(guān)鍵字允許省略變量的類型。
jshell> var x = new HashSet<String>() x ==> [] jshell> x.add("apple") $1 ==> true
在上面的代碼段中,編譯器x可以將的類型推斷為HashSet。
此功能有助于減少樣板代碼并提高可讀性。但是,它有一些局限性:您只能var在方法主體內(nèi)部使用,并且編譯器將在編譯時(shí)推斷類型,因此所有內(nèi)容仍為靜態(tài)類型。
單一源文件啟動(dòng)(Java 11)
以前,當(dāng)您編寫(xiě)了一個(gè)包含一個(gè)文件的簡(jiǎn)單Java程序時(shí),必須先使用編譯文件,javac然后使用來(lái)運(yùn)行它java。在Java 11中,您可以使用一個(gè)命令完成兩個(gè)步驟:
Main.java: public class Main { public static void main(String[] args) { System.out.println("hello world"); } } $ java ./Main.java hello world
對(duì)于僅由一個(gè)Java類組成的簡(jiǎn)單啟動(dòng)程序或?qū)嶒?yàn),此用于啟動(dòng)單個(gè)源文件的功能將使您的生活更輕松。
Switch表達(dá)式(Java 12)
Java 12為我們帶來(lái)了Switch表達(dá)式。這是該表達(dá)式與舊的switch語(yǔ)句有何不同的快速展示。
在老switch語(yǔ)句定義了程序的流程:
jshell> var i = 3 jshell> String s; jshell> switch(i) { ...> case 1: s = "one"; break; ...> case 2: s = "two"; break; ...> case 3: s = "three"; break; ...> default: s = "unknown number"; ...> } jshell> s s ==> "three"
相反,新的switch表達(dá)式返回一個(gè)值:
jshell> var i = 3; jshell> var x = switch(i) { ...> case 1 -> "one"; ...> case 2 -> "two"; ...> case 3 -> "three"; ...> default -> "unknown number"; ...> }; x ==> "three"
總而言之,舊的switch語(yǔ)句用于程序流,新的switch表達(dá)式解析為一個(gè)值。
請(qǐng)注意,這個(gè)新的switch語(yǔ)句是一種映射函數(shù):有一個(gè)輸入(在上述情況下i),有一個(gè)輸出(在此x)。這實(shí)際上是一種模式匹配功能,有助于使Java與哈四年編程原理更加兼容。一個(gè)類似的switch語(yǔ)句已經(jīng)在斯卡拉推出已有一段時(shí)間。
需要注意的幾件事:
代替雙點(diǎn),我們使用箭頭 ->
不需要 break
考慮所有可能的情況時(shí),可以省略默認(rèn)情況
要在Java 12中啟用此功能,請(qǐng)使用 --enable-preview --source 12
多行字符串(Java 13)
您是否曾經(jīng)定義過(guò)長(zhǎng)的多行字符串,例如JSON或XML?到目前為止,您可能已經(jīng)將所有內(nèi)容都?jí)嚎s了一行并使用換行符\n,但這使String更加難以閱讀。Java 13帶有多行字符串!
案例:
public class Main { public static void main(String [] args) { var s = """ { "recipe": "watermelon smoothie", "duration": "10 mins", "items": ["watermelon", "lemon", "parsley"] }"""; System.out.println(s); } }
現(xiàn)在,我們通過(guò)單文件啟動(dòng)運(yùn)行main方法:
java --enable-preview --source 13 Main.java { "recipe": "watermelon smoothie", "duration": "10 mins", "items": ["watermelon", "lemon", "parsley"] }
結(jié)果字符串跨越多行,引號(hào)""保持不變,甚至制表符\t也被保留!
數(shù)據(jù)類:record(Java 14)
在本文的所有新功能中,這可能是我最興奮的功能:最后,Java中有數(shù)據(jù)類!這些類用record關(guān)鍵字聲明,并具有自動(dòng)Getter,構(gòu)造函數(shù)和equals()方法等??傊?,您可以擺脫大量的樣板代碼!
jshell> record Employee (String name, int age, String department) {} | created record Employee jshell> var x = new Employee("Anne", 25, "Legal"); x ==> Employee[name=Anne, age=25, department=Legal] jshell> x.name() $2 ==> "Anne"
Scala對(duì)于案例類具有類似的功能,對(duì)于Kotlin具有數(shù)據(jù)類具有類似的功能。到目前為止,在Java中,許多開(kāi)發(fā)人員都使用Lombok,它提供了許多現(xiàn)在受recordsJava 14啟發(fā)的功能。有關(guān)更多詳細(xì)信息,請(qǐng)參見(jiàn)Baeldung文章。
instanceof 沒(méi)有演員表(Java 14)
Java的早期版本已經(jīng)包含instanceof關(guān)鍵字:
Object obj = new String("hello"); if (obj instanceof String) { System.out.println("String length: " + ((String)obj).length()); }
不幸的部分:首先我們檢查s類型是否為String,然后再次對(duì)其進(jìn)行強(qiáng)制轉(zhuǎn)換以獲取其長(zhǎng)度。
現(xiàn)在使用Java 14,編譯器足夠聰明,可以在instanceof check之后自動(dòng)推斷類型:
Object obj = new String("hello"); if (obj instanceof String mystr) { System.out.println("String length: " + mystr.length()); }
密封的類(Java 15)
使用sealed關(guān)鍵字,您可以限制哪些類可以擴(kuò)展給定的類或接口。這是一個(gè)例子:
public sealed interface Fruit permits Apple, Pear { String getName(); } public final class Apple implements Fruit { public String getName() { return "Apple"; } } public final class Pear implements Fruit { public String getName() { return "Pear"; } }
那么這對(duì)我們有什么幫助呢?好吧,現(xiàn)在您知道有多少個(gè)了Fruits。實(shí)際上,這是朝著完全支持的模式匹配的方向邁出的重要一步,在該模式下,您可以像對(duì)待枚舉一樣對(duì)待類。此sealed功能與switch前面說(shuō)明的新表達(dá)式很好地結(jié)合在一起。
總結(jié)
在過(guò)去的6年中,Java取得了長(zhǎng)足的發(fā)展,此后實(shí)際上已經(jīng)發(fā)布了8個(gè)Java新版本!與其他基于JVM的競(jìng)爭(zhēng)對(duì)手(Scala和Kotlin)相比,所有這些令人敬畏的新功能有助于使Java成為競(jìng)爭(zhēng)選擇。
如果您正在尋找Java 8以后的Java新功能的更多詳細(xì)信息,我可以推薦Andrew的DEV文章和David Csakvari的這篇文章。
以上就是java8到j(luò)ava15的新功能簡(jiǎn)介的詳細(xì)內(nèi)容,更多關(guān)于java 新功能的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
基于指針pointers和引用references的區(qū)別分析
本篇文章介紹了,基于指針pointers和引用references的區(qū)別分析。需要的朋友參考下2013-05-05java教程之java程序編譯運(yùn)行圖解(java程序運(yùn)行)
最近重新復(fù)習(xí)了一下java基礎(chǔ),在使用javap的過(guò)程中遇到了一些問(wèn)題,這里便講講對(duì)于一個(gè)類文件如何編譯、運(yùn)行、反編譯的。也讓自己加深一下印象2014-03-03Spring boot實(shí)現(xiàn)熱部署的兩種方式詳解
這篇文章主要介紹了Spring boot實(shí)現(xiàn)熱部署的兩種方式,這兩種方法分別是使用 Spring Loaded和使用spring-boot-devtools進(jìn)行熱部署,文中給出了詳細(xì)示例代碼和介紹,需要的朋友可以參考學(xué)習(xí),下面來(lái)一起看看吧。2017-04-04Layui前后臺(tái)交互數(shù)據(jù)獲取java實(shí)例
下面小編就為大家分享一篇Layui前后臺(tái)交互數(shù)據(jù)獲取java實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-01-01Logback配置文件這么寫(xiě),還說(shuō)你不會(huì)整理日志?
logback框架會(huì)默認(rèn)加載classpath下命名為logback-spring.xml或logback.xml的配置文件。這篇文章主要介紹了Logback配置文件寫(xiě)法,需要的朋友可以參考下2020-07-07JSP 開(kāi)發(fā)之 releaseSession的實(shí)例詳解
這篇文章主要介紹了JSP 開(kāi)發(fā)之 releaseSession的實(shí)例詳解的相關(guān)資料,需要的朋友可以參考下2017-07-07SpringBoot實(shí)現(xiàn)登錄攔截的示例代碼
如果我們不進(jìn)行登錄攔截的話,即使我們跳過(guò)登錄頁(yè)面直接去訪問(wèn)任意一個(gè)頁(yè)面也能訪問(wèn)成功,那么登錄功能就沒(méi)有意義,同時(shí)也會(huì)存在安全問(wèn)題,本文就來(lái)介紹一下SpringBoot登錄攔截,具有一定的參考價(jià)值,感興趣的可以了解一下2023-09-09SpringBoot項(xiàng)目調(diào)優(yōu)及垃圾回收器的比較詳解
這篇文章主要介紹了SpringBoot項(xiàng)目調(diào)優(yōu)及垃圾回收器的比較詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-04-04