
Java 14 早在 2020 年 3 月就已经发布,虽然不是长久支持版本,但是也带来了不少新功能。
Java 14 官方下载:https://jdk.java.net/archive/
Java 14 官方文档:https://openjdk.java.net/projects/jdk/14/
Java 14 新功能:
- 305: instanceof 类型判断(预览)
- 343: 打包工具(孵化)
- 345: G1 支持 NUMA(非统一内存访问)
- 358: 更有用的 NullPointerExceptions
- 359: Records (预览)
- 361: Switch 表达式 (标准)
- 362: 废弃对 Solaris 和 SPARC 端口支持
- 363: 移除 CMS 垃圾收集器
- 364: macOS 的 ZGC
- 365: Windows 的 ZGC
- 366: 弃用 ParallelScavenge + SerialOld GC 组合
- 367: 删除 Pack200 Tools 和 API
- 368: 文本块 (二次预览)
- 370: Foreign-Memory Access API (Incubator)
- 349: JFR Event Streaming
- 352: Non-Volatile Mapped Byte Buffers
注意: 如果一个功能为预览版,那么在编译和运行时需要打开预览功能。
./javac --enable-preview --release 14 Test.java
./java --enable-preview Test
此文章属于 Java 新特性教程 系列,会介绍 Java 每个版本的新功能,可以点击浏览。
1. JEP 305:instanceof 类型判断(预览)
在 Java 14 之前,使用 instanceof 进行类型判断之后,需要进行对象类型转换后才能使用。
package com.wdbyte;
import java.util.ArrayList;
import java.util.List;
public class Java14BeaforInstanceof {
    public static void main(String[] args) {
        Object obj = new ArrayList<>();
        if (obj instanceof ArrayList) {
            ArrayList list = (ArrayList)obj;
            list.add("www.wdbyte.com");
        }
        System.out.println(obj);
    }
}
而在 Java 14 中,可以在判断类型时指定变量名称进行类型转换,方便了使用。
package com.wdbyte;
import java.util.ArrayList;
public class Java14Instanceof {
    public static void main(String[] args) {
        Object obj = new ArrayList<>();
        if (obj instanceof ArrayList list) {
            list.add("www.wdbyte.com");
        }
        System.out.println(obj);
    }
}
可以看到,在使用 instanceof 判断类型成立后,会自动强制转换类型为指定类型。
输出结果:
[www.wdbyte.com]
这个特性在 Java 14 中还是预览功能,在 Java 16 中正式转正。
2. JEP 343:打包工具(孵化)
在 Java 14 中,引入了打包工具,命令是 jpackage,使用 jpackage 命令可以把 JAR 包打包成不同操作系统支持的软件格式。
jpackage --name myapp --input lib --main-jar main.jar --main-class myapp.Main
常见平台格式如下:
- Linux: debandrpm
- macOS: pkganddmg
- Windows: msiandexe
要注意的是,jpackage 不支持交叉编译,也就是说在 windows 平台上是不能打包成 macOS 或者 Linux 系统的软件格式的。
3. JEP 345: G1 支持 NUMA(非统一内存访问)
G1 收集器现在可以感知 NUMA 内存分配方式,以提高 G1 的性能,可以使用 +XX:+UseNUMA 启用这项功能。
扩展阅读:https://openjdk.java.net/jeps/345
4. JEP 358:更有用的 NullPointerExceptions
NullPointerException 一直都是一个比较常见的异常,但是在 Java 14 之前,如果一行有多个表达式时,这时报了空指针后,单纯的从报错信息来看,可能并不知道是哪个对象为 NULL ,下面是一个演示。
package com.wdbyte;
public class Java14NullPointerExceptions {
    public static void main(String[] args) {
        String content1 = "www.wdbyte.com";
        String content2 = null;
        int length = content1.length() + content2.length();
        System.out.println(length);
    }
}
在 Java 14 之前,从下面的报错中我们只能得到错误出现的行数,但是并不能确定是 conteng1 还是 content2 为 null。
Exception in thread "main" java.lang.NullPointerException
	at com.alibaba.security.astralnet.console.controller.ApiChartsTest.main(Java14NullPointerExceptions.java:8)
但是在 Java 14 中,会清晰的告诉你 because "content2" is null 。
Exception in thread "main" java.lang.NullPointerException: 
	Cannot invoke "String.length()" because "content2" is null
	at com.wdbyte.Java14NullPointerExceptions.main(Java14NullPointerExceptions.java:8)
5. JEP 359:Records (预览)
record 是一种全新的类型,它本质上是一个 final 类,同时所有的属性都是 final 修饰,它会自动编译出 public get hashcode 、equals、toString 等方法,减少了代码编写量。
示例:编写一个 Dog record 类,定义 name 和 age 属性。
package com.wdbyte;
public record Dog(String name, Integer age) {
}
Record 的使用。
package com.wdbyte;
public class Java14Record {
    public static void main(String[] args) {
        Dog dog1 = new Dog("牧羊犬", 1);
        Dog dog2 = new Dog("田园犬", 2);
        Dog dog3 = new Dog("哈士奇", 3);
        System.out.println(dog1);
        System.out.println(dog2);
        System.out.println(dog3);
    }
}
输出结果:
Dog[name=牧羊犬, age=1]
Dog[name=田园犬, age=2]
Dog[name=哈士奇, age=3]
这个功能在 Java 15 中进行二次预览,在 Java 16 中正式发布。
6. JEP 361:Switch 表达式 (标准)
Switch 表达式改进从 Java 12 就已经开始了,Java 12 让 switch 支持了 L-> 语法,Java 13 引入了 yield 关键词用于返回结果,但是在 Java 12 和 13 中功能都是预览版的,而在 Java 14 中,正式转正。
// 通过传入月份,输出月份所属的季节
public static String switchJava12(String month) {
     return switch (month) {
        case "march", "april", "may"            -> "春天";
        case "june", "july", "august"           -> "夏天";
        case "september", "october", "november" -> "秋天";
        case "december", "january", "february"  -> "冬天";
        default -> "month erro";
    };
}
// 通过传入月份,输出月份所属的季节
public static String switchJava13(String month) {
    return switch (month) {
        case "march", "april", "may":
            yield "春天";
        case "june", "july", "august":
            yield "夏天";
        case "september", "october", "november":
            yield "秋天";
        case "december", "january", "february":
            yield "冬天";
        default:
            yield "month error";
    };
}
7. JEP 368:文本块(二次预览)
文本块是 Java 13 引入的语法,在 Java 14 中对其进行了增强。文本块依旧是预览功能,这次更新增加了两个转义符。
- \结尾不换行
- \s表示一个空格
示例:文本块体验
String content = """
        {
            "upperSummary": null,\
            "sensitiveTypeList": null,
            "gmtModified": "2011-08-05\s10:50:09",
        }
         """;
System.out.println(content);
输出结果:
{
    "upperSummary": null,    "sensitiveTypeList": null,
    "gmtModified": "2011-08-05 10:50:09",
}
文本块功能在 Java 15 中正式发布。
其他更新
JEP 362:废弃对 Solaris 和 SPARC 端口支持
从 Java 14 开始,放弃对 Solaris/SPARC, Solaris/x64, 和 Linux/SPARC 端口的支持,放弃一部分开发这势必会加快 Java 整体的开发节奏。
相关阅读:https://openjdk.java.net/jeps/362
JEP 363: 移除 CMS 垃圾收集器
移除对 CMS(Concurrent Mark Sweep) 垃圾收集器的支持,其实早在 Java 9 就开始移除 CMS 垃圾收集器了,只不过在 Java 14 中被正式删除。
JEP 364:macOS 上的 ZGC(实验性)
Java 11 在 Linux 上引入了 Z 垃圾收集器 (ZGC),现在它可以移植到 macOS。
JEP 365:Windows 上的 ZGC(实验性)
Java 11 在 Linux 上引入了 Z 垃圾收集器 (ZGC),现在它可以移植到 Windows 上(版本大于 1803)。
JEP 366:弃用 ParallelScavenge + SerialOld GC 组合
由于使用场景不多,维护工作太大,废弃之。相关阅读:https://openjdk.java.net/jeps/366
JEP 367:删除 Pack200 工具和 API
参考
- https://openjdk.java.net/projects/jdk/14/
- https://openjdk.java.net/jeps/366
- https://openjdk.java.net/jeps/362
- Java 新特性教程