我還在生產(chǎn)玩 JDK7,JDK 15 卻要來了!|新特性嘗鮮
點(diǎn)擊藍(lán)色“程序通事”關(guān)注我喲
加個(gè)“星標(biāo)”,不迷路
自從 JDK9 之后,每年 3 月與 9 月 JDK 都會(huì)發(fā)布一個(gè)新的版本,而2020 年 9 月即將引來 JDK15。
恰巧 IDEA 每四五個(gè)月會(huì)升級(jí)一個(gè)較大的版本,每次升級(jí)之后都會(huì)支持最新版本 JDK 引入的新功能。
這幾天升級(jí)了 IDEA,順便體驗(yàn)了一下 JDK15 的新特性。
雖然我知道你們可能跟我一樣JDK8 都還沒用熟,但是無妨,看看新版本 JDK 來酸一下。

Text Blocks 最終定板
之前版本的 JDK,如果我們需要插入 HTML,XML,SQL 或 JSON 片段,非常麻煩,需要對(duì)里面符號(hào)進(jìn)行各種轉(zhuǎn)義。
所以我每次都會(huì)在其他編輯器將 HTML ,XML 等編輯好,然后直接復(fù)制到 IDEA 中,IDEA 自動(dòng)會(huì)對(duì)這些字符轉(zhuǎn)義。

每次復(fù)制進(jìn)去就變成上圖的效果,如果上面字符再多點(diǎn),閱讀起來就會(huì)更難,并且難以維護(hù)。
所幸 IDEA 提供了一個(gè) Inject Language 功能,我們可以在里面快速方便的編輯。

Java 開發(fā)者也關(guān)注到這個(gè)問題,他們?cè)?JDK13 引入的一個(gè)新的預(yù)覽特性「Text Blocks」,可以使用三引號(hào)將復(fù)雜的字符串賦值,從而讓我們從各種轉(zhuǎn)義中解脫出來,可以更加方便的編輯字符串。
這個(gè)功能在其他語(yǔ)言還是比較常見的,比如 Python 等。
Text Blocks 新功能在 JDK14 再次以預(yù)覽功能引入,最終在 JDK15 成為新版本的正式功能。
下面我們來對(duì)比一下使用 Text Blocks 與之前區(qū)別:



Records (Second Preview)
JDK14 引入一個(gè)新的預(yù)覽特性 record 語(yǔ)法,可以快速創(chuàng)建一個(gè)純數(shù)據(jù)類,并且不用去生成 getter,toString 等。
使用下面的語(yǔ)法就可以快速創(chuàng)建一個(gè)數(shù)據(jù)類:
public?record?Point(int?x,int?y)?{
}
JDK15 是 record 這個(gè)語(yǔ)法的第二次預(yù)覽,這個(gè)版本增加一個(gè)新的功能 「local record」,可以在一個(gè)方法在快速創(chuàng)建一個(gè)類,以便于方法中業(yè)務(wù)邏輯計(jì)算。
在以下示例中,使用本地記錄 MerchantSales 對(duì)商人和每月銷售額的匯總進(jìn)行建模,使用此記錄可提高以下流操作的可讀性:
下面例子的中我們新建一個(gè)類 MerchantSales,然后按照銷售人員對(duì)每月的銷售額匯總排序。
List?findTopMerchants(List?merchants,?int?month) ? {
????//?Local?record
????record?MerchantSales(Merchant?merchant,?double?sales)?{}
????return?merchants.stream()
????????.map(merchant?->?new?MerchantSales(merchant,?computeSales(merchant,?month)))
????????.sorted((m1,?m2)?->?Double.compare(m2.sales(),?m1.sales()))
????????.map(MerchantSales::merchant)
????????.collect(toList());
}
原先如果需要使用這種功能,我們不得不創(chuàng)建一個(gè)內(nèi)部類,后續(xù)可能再也不會(huì)用到,使用 local record就解決這個(gè)尷尬的問題。
除了 ?local record 我們還可以創(chuàng)建 local enums 以及 local interface。
//?local?enums
public?void?organisePeople(List?people) ?{
????enum?Role?{
????????Employee,?Customer,?Both,?None
????}
????HashMap>?peopleByRole?=?new?HashMap<>();
????people.stream()
????????????.filter(Person::isCustomer)
????????????.forEach(person?->?peopleByRole.computeIfAbsent(Role.Customer,?role?->?new?ArrayList<>())
????????????????????.add(person));
????//?其他業(yè)務(wù)邏輯
}
//?local?interface
public?void?localInterface()?{
????interface?MyInterface?{
????????void?doSomething();
????}
????MyInterface?testInterface?=?new?MyInterface()?{
????????@Override
????????public?void?doSomething()?{
????????????System.out.println("Hello?World!");
????????}
????};
????//?其他業(yè)務(wù)邏輯
}
最后使用這個(gè)特性需要注意一點(diǎn),local record , local enums ,local interface 創(chuàng)建都是一個(gè)局部變量,是不能被傳遞其他方法引用。
Pattern Matching for instanceof (Second Preview)
我們應(yīng)該都看到過下面這種代碼:
if?(obj?instanceof?String)?{
????String?str?=?(String)?obj;
????//?use?str
}
上面代碼意圖非常簡(jiǎn)單,當(dāng) obj 對(duì)象是 String 類,就將其強(qiáng)制轉(zhuǎn)換,然后進(jìn)行其他業(yè)務(wù)操作。
這種寫法,類型轉(zhuǎn)換還是比較繁瑣,Pattern Matching for instanceof ?這個(gè)新語(yǔ)法特性,可以幫我們省略這種類型轉(zhuǎn)換動(dòng)作。這是一個(gè)在 JDK14 引入一個(gè)預(yù)覽特性,JDK 15 開始第二次預(yù)覽。
上面的代碼使用 pattern matcher,就可以被修改如下:
if?(obj?instanceof?String?s)?{
????s.contains("T");
}?else?{
????//?編譯錯(cuò)誤
????//s.contains("T");
}
另外如果在 IDEA 中還可以提示我們將代碼轉(zhuǎn)化成 pattern matcher 。

大家應(yīng)該都看過 Effective Java 這本神書吧,里面第八條關(guān)于 Equals 有一個(gè)例子:

使用 pattern matcher 我們就可以使用下面更加清晰的代碼代替:

Sealed Classes (Preview)
Java 中一個(gè)正常普通類/接口允許被其他子類繼承/實(shí)現(xiàn),但是有時(shí)在日常開發(fā)中,我們可能希望只有特定的類才能繼承擴(kuò)展。
現(xiàn)有的 Java 語(yǔ)法中存在一些方法,可以限制子類擴(kuò)展,比如說:我們可以使用 final 修飾類
public?final?class?String
不過這樣之后,我們就沒辦法再繼承這個(gè)類。
其次我們可以限制的類的范圍,比如說不使用 public 修飾類/接口,即使用 default 范圍,這樣只有同一個(gè)包才能繼承/實(shí)現(xiàn)。
interface?DefaultExample?{
}
不過使用這種方式,又很尷尬,這個(gè)類就無法被其他包使用。
為了解決上述問題,JDK 15 引入一個(gè)新的預(yù)覽特性 Sealed Classes,即可以限定類的擴(kuò)展,也可以被外部使用。
public?sealed?class?Shape
????permits?Circle,?Rectangle,?Square?{...}
使用 sealed 修飾之后,Shape 類只能被 Circle,Rectangle,Square繼承,再也不能被其他類繼承。
同時(shí) Shape 的子類存在一些限制,必須使用 final 修飾,表明這個(gè)類無法再被擴(kuò)展:
public?final?class?Circle?extends?Shape?{...}
或者繼續(xù)使用 sealed 表示子類只能被指定類繼承:
public?sealed?class?Rectangle?extends?Shape?
????permits?TransparentRectangle,?FilledRectangle?{...}
public?final?class?TransparentRectangle?extends?Rectangle?{...}
又或者說使用 non-sealed 表明這個(gè)子類不限制子類擴(kuò)展,可以被其他任何類擴(kuò)展實(shí)現(xiàn)。
另外 sealed class 還可以跟上述 record 語(yǔ)法一起使用。
public?sealed?interface?Expr
????permits?ConstantExpr,?PlusExpr,?TimesExpr,?NegExpr?{...}
public?record?ConstantExpr(int?i)???????implements?Expr?{...}
public?record?PlusExpr(Expr?a,?Expr?b)??implements?Expr?{...}
public?record?TimesExpr(Expr?a,?Expr?b)?implements?Expr?{...}
public?record?NegExpr(Expr?e)???????????implements?Expr?{...}
ZGC
ZGC(Z Garbage Collector) 這是一款在 JDK11 引入的的具有實(shí)驗(yàn)性質(zhì)的低延遲的 GC 收集器。
這款 GC 收集器的希望在盡可能對(duì)吞吐量影響不大的前提下,實(shí)現(xiàn)在任意堆內(nèi)存大小都可以把垃圾收集器的停頓時(shí)間限制在十毫秒以內(nèi)的低延遲。
ZGC 經(jīng)過這兩三年的迭代優(yōu)化,終于在 JDK15 中正式引入,標(biāo)志著 ZGC 可以正式應(yīng)用于生產(chǎn)應(yīng)用。
JDK15 中默認(rèn)虛擬機(jī)還是 G1,如果需要使用 ZGC,需要在啟動(dòng)參數(shù)中加入如下參數(shù):
-XX:+UseZGC?command-line?
最后
本來這篇文章是準(zhǔn)備寫下 IDEA 2020.2 新版本特性,順帶介紹一下 JDK15 新特性的。
可是沒想到寫著寫著,JDK15 相關(guān)的篇幅就過長(zhǎng)了,所以就單獨(dú)拿出來了。
最后,最后,JDK 都發(fā)布到 15 了,而我卻還在用 JDK 7 ,真是個(gè)悲傷的故事,逃了逃了!

我是樓下小黑哥,每天學(xué)習(xí)一點(diǎn)點(diǎn),成長(zhǎng)億點(diǎn)點(diǎn)!!
參考鏈接
https://openjdk.java.net/projects/jdk/15/
- END -老大吩咐的可重入分布式鎖,終于完美的實(shí)現(xiàn)了~
造了一個(gè) Redis 分布鎖的輪子,沒想到還學(xué)到這么多東西?。?!
MySQL 可重復(fù)讀,差點(diǎn)就讓我背上了一個(gè) P0 事故!
