我还在生产玩 JDK7,JDK 15 却要来了!|新特性尝鲜
自從 JDK9 之后,每年 3 月與 9 月 JDK 都會(huì)發(fā)布一個(gè)新的版本,而2020 年 9 月即將引來(lái) JDK15。
恰巧 IDEA 每四五個(gè)月會(huì)升級(jí)一個(gè)較大的版本,每次升級(jí)之后都會(huì)支持最新版本 JDK 引入的新功能。
這幾天升級(jí)了 IDEA,順便體驗(yàn)了一下 JDK15 的新特性。
雖然我知道你們可能跟我一樣JDK8 都還沒(méi)用熟,但是無(wú)妨,看看新版本 JDK 來(lái)酸一下。
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),閱讀起來(lái)就會(huì)更難,并且難以維護(hù)。
所幸 IDEA 提供了一個(gè) Inject Language 功能,我們可以在里面快速方便的編輯。
Java 開發(fā)者也關(guān)注到這個(gè)問(wèn)題,他們?cè)?JDK13 引入的一個(gè)新的預(yù)覽特性「Text Blocks」,可以使用三引號(hào)將復(fù)雜的字符串賦值,從而讓我們從各種轉(zhuǎn)義中解脫出來(lái),可以更加方便的編輯字符串。
這個(gè)功能在其他語(yǔ)言還是比較常見(jiàn)的,比如 Python 等。
Text Blocks 新功能在 JDK14 再次以預(yù)覽功能引入,最終在 JDK15 成為新版本的正式功能。
下面我們來(lái)對(duì)比一下使用 Text Blocks 與之前區(qū)別:
HtmlSQLJSRecords?(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<Merchant>?findTopMerchants(List<Merchant>?merchants,?int?month)?{//?Local?recordrecord?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è)尷尬的問(wèn)題。
除了 ?local record 我們還可以創(chuàng)建 local enums 以及 local interface。
//?local?enums public?void?organisePeople(List<Person>?people)?{enum?Role?{Employee,?Customer,?Both,?None}HashMap<Role,?List<Person>>?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()?{@Overridepublic?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)該都看到過(guò)下面這種代碼:
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)該都看過(guò) Effective Java 這本神書吧,里面第八條關(guān)于 Equals 有一個(gè)例子:
使用 pattern matcher 我們就可以使用下面更加清晰的代碼代替:
Sealed?Classes?(Preview)
Java 中一個(gè)正常普通類/接口允許被其他子類繼承/實(shí)現(xiàn),但是有時(shí)在日常開發(fā)中,我們可能希望只有特定的類才能繼承擴(kuò)展。
現(xiàn)有的 Java 語(yǔ)法中存在一些方法,可以限制子類擴(kuò)展,比如說(shuō):我們可以使用 final 修飾類
public?final?class?String不過(guò)這樣之后,我們就沒(méi)辦法再繼承這個(gè)類。
其次我們可以限制的類的范圍,比如說(shuō)不使用 public 修飾類/接口,即使用 default 范圍,這樣只有同一個(gè)包才能繼承/實(shí)現(xiàn)。
interface?DefaultExample?{ }不過(guò)使用這種方式,又很尷尬,這個(gè)類就無(wú)法被其他包使用。
為了解決上述問(wèn)題,JDK 15 引入一個(gè)新的預(yù)覽特性 Sealed Classes,即可以限定類的擴(kuò)展,也可以被外部使用。
public?sealed?class?Shapepermits?Circle,?Rectangle,?Square?{...}使用 sealed 修飾之后,Shape 類只能被 Circle,Rectangle,Square繼承,再也不能被其他類繼承。
同時(shí) Shape 的子類存在一些限制,必須使用 final 修飾,表明這個(gè)類無(wú)法再被擴(kuò)展:
public?final?class?Circle?extends?Shape?{...}或者繼續(xù)使用 sealed 表示子類只能被指定類繼承:
public?sealed?class?Rectangle?extends?Shape?permits?TransparentRectangle,?FilledRectangle?{...} public?final?class?TransparentRectangle?extends?Rectangle?{...}又或者說(shuō)使用 non-sealed 表明這個(gè)子類不限制子類擴(kuò)展,可以被其他任何類擴(kuò)展實(shí)現(xiàn)。
另外 sealed class 還可以跟上述 record 語(yǔ)法一起使用。
public?sealed?interface?Exprpermits?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)過(guò)這兩三年的迭代優(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?最后
本來(lái)這篇文章是準(zhǔn)備寫下 IDEA 2020.2 新版本特性,順帶介紹一下 JDK15 新特性的。
可是沒(méi)想到寫著寫著,JDK15 相關(guān)的篇幅就過(guò)長(zhǎng)了,所以就單獨(dú)拿出來(lái)了。
最后,最后,JDK 都發(fā)布到 15 了,而我卻還在用 JDK 7 ,真是個(gè)悲傷的故事,逃了逃了!
參考鏈接
https://openjdk.java.net/projects/jdk/15/
有道無(wú)術(shù),術(shù)可成;有術(shù)無(wú)道,止于術(shù)
歡迎大家關(guān)注Java之道公眾號(hào)
好文章,我在看??
總結(jié)
以上是生活随笔為你收集整理的我还在生产玩 JDK7,JDK 15 却要来了!|新特性尝鲜的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 小圈子.大社交:利用圈子引爆流行
- 下一篇: 2020 大厂研发岗薪酬排名出炉,看完我