<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          我還在生產(chǎn)玩 JDK7,JDK 15 卻要來了!|新特性嘗鮮

          共 4630字,需瀏覽 10分鐘

           ·

          2020-08-02 07:21

          點(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,XMLSQLJSON 片段,非常麻煩,需要對(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ū)別:

          Html
          SQL
          JS

          Records (Second Preview)

          JDK14 引入一個(gè)新的預(yù)覽特性 record 語(yǔ)法,可以快速創(chuàng)建一個(gè)純數(shù)據(jù)類,并且不用去生成 gettertoString 等。

          使用下面的語(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 事故!




          瀏覽 42
          點(diǎn)贊
          評(píng)論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          評(píng)論
          圖片
          表情
          推薦
          點(diǎn)贊
          評(píng)論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  国产在线小视频 | 亚洲第一综合 | 最新黄色在线网站 | 伊人青青在线观看视频 | 99热热久久 |