9 月 19 日,Oracle 宣布 Java 21 正式发布。 自从 Java 改为 6 个月发布一个版本以来,此为如期交付的第 12 个版本。
根据规划,Java 21 是长期支持版本,Oracle 将至少提供 8 年支持。根据客户反馈和 Java 生态系统使用情况,Oracle 还宣布将 Java 11 的支持至少延长至 2032 年 1 月,也就是至少再提供 8 年的支持与更新。
从 Java 11 到 Java 21 的正式发布,一共修复了 24,196 个 JIRA 问题,其中 17,288 个是由 Oracle 的工作人员完成的,而 6,908 个是由个人开发人员和其他组织的开发人员贡献的。
从下图可以看到,中国的腾讯、阿里巴巴、华为和龙芯做出了不少贡献。
盘点 Java 21 的新特性
除了大量的性能、稳定性和安全性方面的更新,Java 21 还包括 15 个 JEP(JDK Enhancement Proposal),其中包括 6 个预览特性和 1 个孵化特性。
这些 JEP 可以分为六类。下面具体来看看一下 Java 21 引入的部分新特性。
(一)Project Amber
JEP 430: String Templates (Preview)
JEP 430:字符串模板(预览特性)
该 JEP 的目标是:
对于需要在运行时求值的字符串,简化其表达方式;
对于既包含文本(不管是单行文本还是多行文本)又包含表达式的复杂表达式,提高其可读性;
有的 Java 程序需要根据用户提供的值来构建字符串并传递给其他系统,字符串模板支持对模板和其中嵌入的表达式的值进行验证和转换,从而提高这类程序的安全性;
允许 Java 库定义字符串模板中使用的格式化语法,保持灵活性;
对于接受 Java 之外的语言(如 SQL、XML、JSON)编写的字符串的 API,使之用起来更加方便;
支持从文字文本和被嵌入的表达式计算非字符串值,而无需通过中间字符串表示进行转换。
String name = "Joan";
String info = STR."My name is \{name}"; // 模板表达式
assert info.equals("My name is Joan"); // 返回true
这里的 STR 是模板处理器,它是 Java 引入的一个 public static final 的字段,会被自动引入到每个 Java 源代码文件中。
因为这是个实验特性,所以编译和运行的时候要添加相关的参数。例如,假设代码保存在 Test.java 文件中,则编译运行方式为:
javac --release 21 --enable-preview Test.java
java --enable-preview Test
该特性可以提高 Java 程序的可读性和可维护性,编写起来也更方便了。
JEP 440: Record Patterns
JEP 440:record 模式
该特性将模式匹配扩展到可以解构 record 类的实例,从而实现更复杂的数据查询。添加嵌套模式,使数据查询可以更方便地组合起来。
在 Java 16 中,要提取一个 record 实例的数据,需要这样操作:
record Point(int x, int y) {}
static void printSum(Object obj) {
if (obj instanceof Point p) {
int x = p.x();
int y = p.y();
System.out.println(x+y);
}
}
而在 Java 21 中,可以直接这样实现:
record Point(int x, int y) {}
static void printSum(Object obj) {
if (obj instanceof Point p) {
int x = p.x();
int y = p.y();
System.out.println(x+y);
}
}
而在 Java 21 中,可以直接这样实现:
static void printSum(Object obj) {
if (obj instanceof Point(int x, int y)) {
System.out.println(x+y);
}
}
JEP 441: Pattern Matching for switch
JEP 441:用于 switch 的模式匹配
该特性为 switch 表达式和语句引入了模式匹配。在 switch 中支持对表达式进行多个模式的测试,每个模式都有特定的操作,从而可以简洁而安全地表示复杂的面向数据的查询。
我们可能希望使用模式来针对同一变量测试多种可能性,并在每种情况下执行特定的操作。例如:
static String formatter(Object obj) {
String formatted = "unknown";
if (obj instanceof Integer i) {
formatted = String.format("int %d", i);
} else if (obj instanceof Long l) {
formatted = String.format("long %d", l);
} else if (obj instanceof Double d) {
formatted = String.format("double %f", d);
} else if (obj instanceof String s) {
formatted = String.format("String %s", s);
}
return formatted;
}
而在 Java 21 中,可以这样实现:
static String formatterPatternSwitch(Object obj) {
return switch (obj) {
case Integer i -> String.format("int %d", i);
case Long l -> String.format("long %d", l);
case Double d -> String.format("double %f", d);
case String s -> String.format("String %s", s);
default -> obj.toString();
};
}
在 Java 21 之前,如果 switch 后面的表达式的值为 null,则会抛出 NullPointerException。
为安全起见,我们会在 switch 语句之外执行检测。
static void testFooBarOld(String s) {
if (s == null) {
System.out.println("Oops!");
return;
}
switch (s) {
case "Foo", "Bar" -> System.out.println("Great");
default -> System.out.println("Ok");
}
}
现在可以这样处理:
static void testFooBarNew(String s) {
switch (s) {
case null -> System.out.println("Oops");
case "Foo", "Bar" -> System.out.println("Great");
default -> System.out.println("Ok");
}
}
JEP 443: Unnamed Patterns and Variables (Preview)
JEP 443:无名模式和无名变量(预览特性)
无名模式可以在不指定 record 组件的名称或类型的情况下匹配组件,无名变量可以用于初始化,但不能被使用。它们都用下划线字符表示。
例如:
int acc = 0;
for (Order _ : orders) {
if (acc < LIMIT) {
... acc++ ...
}
}
再如下例:
String s = ...
try {
int i = Integer.parseInt(s);
... i ...
} catch (NumberFormatException _) {
System.out.println("Bad number: " + s);
}
JEP 445: Unnamed Classes and Instance Main Methods (Preview)
JEP 445:无名的类和实例 main 方法(预览特性)
方便学生学习,无需理解很多为编写大型程序而设计的语言特性即可上手。
之前,要写个最简单的程序也要写这么多代码:
public class HelloWorld {
public static void main(String[] args) {
System.out.println("Hello, World!");
}
}
现在可以简化为这样,似乎在向Python,PHP等脚本语言看齐:
class HelloWorld {
void main() {
System.out.println("Hello, World!");
}
}
Java 甚至还引入了无名的类,将类的声明隐藏掉:
void main() {
System.out.println("Hello, World!");
}
(二)Project Loom
JEP 444: Virtual Threads
JEP 444:虚拟线程
这是很重要的一个特性,将虚拟线程引入到了Java平台中。虚拟线程是轻量级线程,大大降低了编写、维护和观察高吞吐量并发应用程序的工作量。
该特性的目标是:
使采用简单的每个请求一个线程的方式编写的服务器应用程序能够充分利用硬件资源进行扩展;
使现有的使用 java.lang.Thread API 的代码能够以最小的修改支持虚拟线程;
使现有的 JDK 工具能够轻松进行虚拟线程的故障排除、调试和性能分析。
虚拟线程是 java.lang.Thread 类的一个实例,它在底层的操作系统线程上运行 Java 代码,但并不会在整个生命周期中占用这个操作系统线程。这意味着许多虚拟线程可以在同一个操作系统线程上运行其 Java 代码,有效地共享该线程。平台线程会独占一个宝贵的操作系统线程,虚拟线程则不会。虚拟线程的数量可以远远大于操作系统线程的数量。虚拟线程是由 JDK 提供的轻量级线程实现,而不是由操作系统提供的。
例如,为每个请求创建一个新的虚拟线程来进行处理的代码:
void handle(Request request, Response response) {
var url1 = ...
var url2 = ...
try (var executor = Executors.newVirtualThreadPerTaskExecutor()) {
var future1 = executor.submit(() -> fetchURL(url1));
var future2 = executor.submit(() -> fetchURL(url2));
response.send(future1.get() + future2.get());
} catch (ExecutionException | InterruptedException e) {
response.fail(e);
}
}
String fetchURL(URL url) throws IOException {
try (var in = url.openStream()) {
return new String(in.readAllBytes(), StandardCharsets.UTF_8);
}
}
JEP 446: Scoped Values (Preview)
JEP 446:作用域值(预览特性)
引入了作用域值,这些值可以安全高效地共享给方法,而不需要使用方法参数。特别是在使用大量虚拟线程时,它们优于线程局部变量。它是 ScopedValue
例如:
final static ScopedValue<...> V = ScopedValue.newInstance();
// In some method
ScopedValue.where(V,
) .run(() -> { ... V.get() ... call methods ... });
// In a method called directly or indirectly from the lambda expression
... V.get() ...
JEP 453: Structured Concurrency (Preview)
JEP 453:结构化并发(预览特性)
引入结构化并发 API,简化并发编程。结构化并发将在不同线程中运行的相关任务组视为一个工作单元,从而简化错误处理和取消操作,进而可靠性,并增强可观测性。结构化并发 API 的主要类是位于 java.util.concurrent 包中的 StructuredTaskScope。
(三)Project Panama
JEP 442: Foreign Function & Memory API (3rd Preview)
JEP 442:外部的函数和内存 API(第 3 次预览)
Foreign Function & Memory API (FFM API) 位于 java.lang.foreign 包中,它定义了一系列类和接口,使Java程序能够与Java运行时之外的代码和数据进行互操作。通过有效地调用外部函数(即JVM之外的代码),并安全地访问外部内存(即并非由JVM管理的内存),该 API 使 Java 程序能够调用本地库并处理本地数据,而不会出现 JNI 的脆弱性和危险性。
JEP 448: Vector API (6th Incubator)
JEP 448:向量 API(第6次 孵化特性)
引入VectorSpecies
(四)核心类库
JEP 431: Sequenced Collections
JEP 431:有顺序的集合
引入新的接口 SequencedCollection
(五)性能更新
JEP 439: Generational ZGC
JEP 439:分代 ZGC
扩展 ZGC,使之在不同的代中维护年轻的对象和年老的对象,从而提高应用程序的性能。这将允许 ZGC 更频繁地收集年轻对象,因为它们更容易早些时候被回收。
JEP 452: Key Encapsulation Mechanism API
JEP 452:密钥封装机制 API
引入了一个新类 KEM,用于密钥封装机制,这是一种使用公钥加密来保护对称密钥的加密技术。
(六)维护与弃用
JEP 449: Deprecate the 32-bit x86 Port for Removal
JEP 449:弃用 32 位 x86 平台构建版本,将来移除
弃用 Windows 32 位 x86 构建版本,并计划在未来的版本中移除。
JEP 451: Prepare to Disallow the Dynamic Loading of Agents
JEP 451:准备禁止动态加载代理
在将代理动态加载到运行中的 JVM 时发出警告。这些警告旨在让用户有所准备,以便在将来的发布版本中,默认禁止动态加载代理。在任何发布版本中,通过服务工具(serviceability tool)在启动时加载代理不会引发警告。
Java 21 的新特性就介绍到这里,赶快下载体验吧!
JDK下载地址:
https://www.oracle.com/java/technologies/downloads/
作者:万能的大雄
本文为 @ 场长 创作并授权 21CTO 发布,未经许可,请勿转载。
内容授权事宜请您联系 webmaster@21cto.com或关注 21CTO 公众号。
该文观点仅代表作者本人,21CTO 平台仅提供信息存储空间服务。