天天看點

JDK15正式釋出,劃時代的ZGC同時宣布轉正✍前言✍正文✍總結

你發任你發,我用Java8。本文已被 https://www.yourbatman.cn 收錄,裡面一并有Spring技術棧、MyBatis、JVM、中間件等小而美的專欄供以免費學習。關注公衆号【BAT的烏托邦】逐個擊破,深入掌握,拒絕淺嘗辄止。
JDK15正式釋出,劃時代的ZGC同時宣布轉正✍前言✍正文✍總結

✍前言

2020年9月15日,JDK15正式釋出,可謂如約而至。按照Java SE的發展路線圖,JDK14自此停止更新。值得注意的是JDK15并非LTS版本,Oracle官方對Java SE的支援路線圖如下:

JDK15正式釋出,劃時代的ZGC同時宣布轉正✍前言✍正文✍總結

JDK8的擴充支援時間超過了JDK11,Oracle你是認真的嗎?開個玩笑~

那麼自Java11之後,哪個版本才是LTS版本呢?Oracle官方并沒給出具體參考路線圖,但可參考OpenJDK的這張:

JDK15正式釋出,劃時代的ZGC同時宣布轉正✍前言✍正文✍總結

可以看到JDK17将是下一個LTS版本,預計發版日期是2021年9月份。當然喽這隻是OpenJDK的發版線路圖,并不代表Oracle官方,是以僅供參考,不過一般八九不離十。

小貼士:OpenJDK和Oracle JDK自從JDK11後,就共享了絕大部分代碼了,節奏基本保持一緻。

從JDK9之後,Oracle采用了新的釋出周期:每6個月釋出一個版本,每3年釋出一個LTS版本。JDK14是繼JDK9之後釋出的第四個版本, 該版本為非LTS版本,最新的LTS版本為JDK11。因為是小鹿快跑,快速疊代,是以此處解釋下這兩個詞:孵化器子產品(Incubator)和預覽特性(Preview)。

孵化器子產品(孵化版/實驗版)

尚未定稿的API/工具,主要用于從Java社群收集使用回報,穩定性無保障,後期有較大可能性移除

預覽特性(預覽版)

規格已成型,實作已确定,但還未最終定稿。這些特性還是存在被移除的可能性,但一般來說最後都會被固定下來。

✍正文

JDK15是Java SE平台的第15個版本,由JSR 390在Java社群程序中指定。

OpenJDK 15是9-15釋出的,Oracle同步跟上。其它廠商的對應JDK版本也會随後跟上

該版本共提供14個新特性,通過這些JEP來表示,截圖如下:

JDK15正式釋出,劃時代的ZGC同時宣布轉正✍前言✍正文✍總結

下面針對其中對開發者日常程式設計關系較大的特性拉出來解釋,并給出對應的使用示例(其實就是JEP 378喽)。

JDK14新特性回顧

老規矩,在進行JDK15的新特性介紹之前,先回顧下JDK14的主要特性有哪些。JDK 14于2020年3月17日釋出。

一、Switch表達式

新的Switch表達式其實早在JDK 12、13中都已存在了,但隻是預覽版,到了JDK 14就徹底變為穩定版了,可以放心商用。

小貼士:預覽版特性是有可能在後續版本中被移除的,但穩定版後幾乎不可能被移除

switch新的表達式有兩個顯著的特點:

  • 支援箭頭表達式傳回
  • 支援yield和return傳回值。

1、箭頭表達式傳回

JDK14之前寫法:

private static void printLetterCount(DayOfWeek dayOfWeek){
    switch (dayOfWeek) {
        case MONDAY:
        case FRIDAY:
        case SUNDAY:
            System.out.println(6);
            break;
        case TUESDAY:
            System.out.println(7);
            break;
        case THURSDAY:
        case SATURDAY:
            System.out.println(8);
            break;
        case WEDNESDAY:
            System.out.println(9);
            break;
    }
}           

要點:break可千萬别忘記寫,否則就是個大bug,并且還比較隐蔽,定位起來稍顯困難。

JDK14等效的新寫法:

private static void printLetterCount(DayOfWeek dayOfWeek){
    switch (dayOfWeek) {
        case MONDAY, FRIDAY, SUNDAY -> System.out.println(6);
        case TUESDAY                -> System.out.println(7);
        case THURSDAY, SATURDAY     -> System.out.println(8);
        case WEDNESDAY              -> System.out.println(9);
    }
}           

可明顯看到新寫法不需要一個個break了,從文法層面規避了我們犯錯的可能性。

2、yield傳回

private static int getLetterCount(DayOfWeek dayOfWeek){
    int letterCount;
    switch (dayOfWeek) {
        case MONDAY:
        case FRIDAY:
        case SUNDAY:
            letterCount = 6;
            break;
        case TUESDAY:
            letterCount = 7;
            break;
        case THURSDAY:
        case SATURDAY:
            letterCount = 8;
            break;
        case WEDNESDAY:
            letterCount = 9;
            break;
        default:
            throw new IllegalStateException("非法: " + dayOfWeek);
    }
    return letterCount;
}           
private static int getLetterCount(DayOfWeek dayOfWeek){
    return switch (dayOfWeek) {
        case MONDAY, FRIDAY, SUNDAY -> 6;
        case TUESDAY                -> 7;
        case THURSDAY, SATURDAY     -> 8;
        case WEDNESDAY              -> 9;
    };
}           

使用箭頭操作符操作效果立竿見影。當然,你還可以使用

yield

關鍵字傳回:

private static int getLetterCount(DayOfWeek dayOfWeek){
    return switch (dayOfWeek) {
        case MONDAY  -> 6;
        default      -> {
            int letterCount = dayOfWeek.toString().length();
            yield letterCount;
        }
    };
}           

二、instanceof的模式比對(預覽)

該功能在JDK14中處理預覽版。

public static void main(String[] args) {
    Object o = "hello world";
    if(o instanceof String ){
        String str = String.class.cast(o);
        System.out.println(str);
    }
}           
public static void main(String[] args) {
    Object o = "hello world";
    // 屁股裡直接可寫個變量名,不再需要強轉了
    if(o instanceof String str){
        System.out.println(str);
    }
}

再如:
if (obj instanceof String s && s.length() > 5) {
    s.contains(..)
}           

如果你運作時有如下錯誤:

java: instanceof 中的模式比對 是預覽功能,預設情況下禁用。
  (請使用 --enable-preview 以啟用 instanceof 中的模式比對)           

那是因為此功能是預覽特性,需要你主動開啟,如下:

JDK15正式釋出,劃時代的ZGC同時宣布轉正✍前言✍正文✍總結

注意:此特性在JDK15中依舊為預覽版。

三、實用的NullPointerException

略。

四、Record(預覽)

Java年紀太大,文法不夠新潮,有時候确實太麻煩,是以有了Record的出現:幹掉那些get/set、toString、equals等方法。

public record Person(String name,Integer age) {
}

public static void main(String[] args) {
    Person person= new Person("YourBatman", 18);
    System.out.println(person);
    System.out.println(person.name());
    System.out.println(person.age());
}           

運作程式,結果列印:

Person[name=YourBatman, age=18]
YourBatman
18           

五、文本塊Text Blocks(二次預覽)

這個特性可是非常好用,它屬于二次預覽:已在JDK 13預覽過一次。

public static void main(String[] args) {
    String html = """
          <html>
              <body>
                  <p>hello world</p>
              </body>
          </html>
          """;
    String query = """
           SELECT * from USER
           WHERE `id` = 1
           ORDER BY `id`, `name`;
           """;
}           

在JDK13中,這種是有換行的。在JDK14中,可以加上一個符号讓其不讓換行:

public static void main(String[] args) {
    String query = """
           SELECT * from USER \
           WHERE `id` = 1 \
           ORDER BY `id`, `name`;\
           """;
    System.out.println(query);
}           

運作程式,輸出(可以看到展示為一行了):

SELECT * from USER WHERE `id` = 1 ORDER BY `id`, `name`;           

注意:此特性在JDK15中已經為正式版。

六、删除CMS垃圾收集器

這款著名的垃圾回收器從這個版本就徹底被删除了。JDK9開始使用G1作為預設的垃圾回收器(JDK11中ZGC開始嶄露頭角),就已經把CMS标記為過期了,在此版本正式删除。

七、ZGC垃圾回收器(實驗)

革命性的ZGC:任意堆大小(TB級别)都能保證延遲在10ms以内,是以低延遲為首要目标的一款垃圾回收器。

在JDK14之前,ZGC隻能用于Linux上,現在也可使用在windows上了

注意:此特性在JDK15中已經為正式版(JDK11開始出現)。

JDK15新特性

有了JDK14新特性回顧做鋪墊,再來了解JDK15的新特性就友善很多了。

特别說明:運作JDK15需要IDEA 2020.2才能支援哦(JDK14要求IDEA 2020.1),然後關于IDEA 2020.2的使用教程(新特性),請移步我公衆号前面發的這篇文章:

IntelliJ IDEA 2020.2正式釋出,諸多亮點總有幾款能助你提效

一、文本塊Text Blocks

Text Blocks首次是在JDK 13中以預覽功能出現的,然後在JDK 14中又預覽了一次,終于在JDK 15中被确定下來,可放心使用了(使用示例請參考文上)。

二、ZGC轉正

ZGC是Java 11引入的新的垃圾收集器(JDK9以後預設的垃圾回收器是G1),經過了多個實驗階段,自此終于成為正式特性。

ZGC是一個重新設計的并發的垃圾回收器,可以極大的提升GC的性能。支援任意堆大小而保持穩定的低延遲(10ms以内),性能非常可觀。

打開方式:使用-XX:+UseZGC指令行參數打開,相信不久的将來它必将成為預設的垃圾回收器。

三、Shenandoah轉正

怎麼形容Shenandoah和ZGC的關系呢?異同點大概如下:

  • 相同點:性能幾乎可認為是相同的
  • 不同點:ZGC是Oracle JDK的,根正苗紅。而Shenandoah隻存在于OpenJDK中,是以使用時需注意你的JDK版本

打開方式:使用-XX:+UseShenandoahGC指令行參數打開。

四、删除Nashorn JavaScript Engine

Nashorn是在JDK提出的腳本執行引擎,早在JDK11就已經把它标記為過期了,JDK15完全移除。

在JDK11中取以代之的是GraalVM。GraalVM是一個運作時平台,它支援Java和其他基于Java位元組碼的語言,但也支援其他語言,如JavaScript,Ruby,Python或LLVM。性能是Nashorn的2倍以上。

五、CharSequence新增isEmpty預設方法

啥都不說,源碼一看便知:

@since 15
default boolean isEmpty() {
    return this.length() == 0;
}           

String實作了CharSequence接口的,這應該地球人都知道吧。

更新建議

自己玩玩就行,畢竟不是LTS版本。

但是,雖然說僅限于自己玩玩就行,但不代表就沒有關注的意義哈。還是那個道理,如果JDK12、13、14、15...都不關注些的話,到時候突然來個JDK17的LTS版本,接受起來就會稍顯困難。

✍總結

JDK15整體來看新特性方面并不算很亮眼,它主要是對之前版本預覽特性的功能做了确定,如文本塊、ZGC等,這麼一來我們就可以放心大膽的使用啦。

半年一次的發版速度真心學不動了,不過還好我有我的堅持:你發任你發,我用Java8。

公衆号背景回複:JDK15,一鍵打包擷取IDEA2020.2.2 + JDK15安裝包(mac + windows)
✔推薦閱讀: