Spring,作为 Java EE 的事实规范,在2022年11月16日发布了最新的 6.0.0 GA 版本。这个版本是框架后续新生代的初始版本,拥抱持续创新的 OpenJDK 和 Java 生态。新的版本以 Java 17+ 作为 baseline,并迁移至 Jakarta EE 9+(即,使用 jakarta 命名空间)。

而在基础设施方面,6.0 首次引入了 AOT 转换,并为 Spring 应用程序上下文提供了相应的 AOT 处理支持。这为 Spring Boot 3 的 GraalVM 原生镜像提供了支持。原生镜像的启动速度非常快,并且能减少 Java 应用程序占用的内存。此外,新版本中支持虚拟线程,虚拟线程是轻量级的线程,能显著减少写入、维护的开销,在并发应用中有较高的吞吐量。

其中很重要的一点是,新的 Spring 6.0 只支持 Java 17+ 了,并且在 Spring 相关的博客中也建议大家升级 JDK 到 17。

JDK LTS

下表是 Oracle 官方提供的 JDK 支持计划:

那为什么是 JDK 17 呢?首先,在 JDK 8 之后只有 JDK 11 和 17 是 LTS(长期维护)版本,而实际上 11 又被大家公认为是过渡版本,对于 JDK 17,Oracle 官宣会提供支持到 2029 年,这给了业界一个相当长的期许,终于可以考虑替换已经诞生 8 年且在2019年1月已经停止公开更新的 JDK 8 了。

开发者关心的功能升级

下面我们看一下 JDK 从 8 升级到 17 的过程中,有哪些另开发者心动的功能呢?

接口私有方法(JDK9)

Java 8 支持在接口中编写默认(default)方法,而从 Java 9 开始,可以在接口中包含私有方法。私有接口方法不能是抽象的。私有方法只能在接口内部使用:

public interface CustomCalculator {
default int addEvenNumbers(int... nums) {
return add(n -> n % 2 == 0, nums);
} default int addOddNumbers(int... nums) {
return add(n -> n % 2 != 0, nums);
} private int add(IntPredicate predicate, int... nums) {
return IntStream.of(nums).filter(predicate).sum();
}
}

本地变量类型推断(JDK10)

在 Java 10 之前版本中,当我们定义局部变量时,需要在赋值的左侧提供显式类型,并在赋值的右边提供实现类型:

Person mike = new Person("Mike");

在 Java 10 以后,可以用下面的方式:

var john = new Person("john");
var doe = new Person("Doe"); // JDK 9 中提供了集合类型的新方法
var persons = List.of(john,doe); // var 也可用于 for
for (var person : persons) {
System.out.println(person.name);
}

虽然我们写代码的时候方便了,但其实这是 JDK 提供的语法糖,在编译成 class 文件时,这些 var 还是会用实际的类型替换。

HTTP Client(JDK11)

java.net.http 包中的 HttpClient 最初在 JDK 9 中提供,后来在 JDK 10 升级,在 JDK 11 终于稳定成为标准功能,同时支持 HTTP/1.1 和 HTTP/2。下面是用 HttpClient 发一个 GET 请求的例子:

HttpClient httpClient = HttpClient.newBuilder()
.version(HttpClient.Version.HTTP_1_1)
.connectTimeout(Duration.ofSeconds(10))
.build(); HttpRequest request = HttpRequest.newBuilder()
.GET()
.uri(URI.create("https://httpbin.org/get"))
.setHeader("User-Agent", "Java 11 HttpClient Bot")
.build(); HttpResponse<String> response =
httpClient.send(request, HttpResponse.BodyHandlers.ofString()); HttpHeaders headers = response.headers();
headers.map().forEach((k, v) -> System.out.println(k + ":" + v)); System.out.println(response.statusCode());
System.out.println(response.body());

Switch 表达式(JDK14)

Switch 表达式是指可以通过 switch 进行赋值,在 Java 12 和 13 中陆续提供了功能预览,而在 Java 14 中成为标准功能。看下面的示例:

private static int getValueViaYield(String mode) {
int result = switch (mode) {
case "a", "b":
yield 1; //使用 yield 提供返回值
case "c":
yield 2;
case "d", "e", "f":
// do something here...
System.out.println("Supports multi line block!");
yield 3;
default:
yield -1;
};
return result;
}

也可以通过箭头函数直接返回,但是这两种方式不能混用:

private static int getValueViaArrow(String mode) {
int result = switch (mode) {
case "a", "b" -> 1;
case "c" -> 2;
case "d", "e", "f" -> {
// do something here...
System.out.println("Supports multi line block!");
yield 3;
}
default -> -1;
};
return result;
}

文本块(JDK15)

文本块功能最开始在 Java 13 和 14 中提供预览,最终在 Java 15 成型,有了这个功能,定义 HTML、SQL 语句或者 JSON,都会更加方便:

String java15DefineHtml = """
<html>
<body>
<p>Welcome to my blog</p>
</body>
</html>
""";
String java15DefineJson = """
{
"name":"世开Coding",
"type":"blog",
"URL":"https://blog.abmcode.com"
}
""";

instanceof 的模式匹配(JDK16)

在 JDK 16 以前,我们用 instanceof 是这样的:

if (url instanceof String) {
String s = (String) url;
if (s.length() > 7) {
if (s.equalsIgnoreCase("Java Development Kit 16")) {
//...
}
}
}

那 JDK 16 之后呢,是这样的,是不是瞬间清爽了:

// str 变量可以在后续逻辑中直接使用
if (url instanceof String str && str.length() > 5 && str.equalsIgnoreCase("Java Development Kit 16")){
// ...
}

Record(JDK16)

Java 被人诟病的一个原因是需要编写太多的模板代码,在新建一个 POJO 时,我们需要提供 getters,setters。Lombok 通过 IDE 工具简化了这个过程,而 Java 16 引入的 Record 类型也同样减少了这种类型的样板代码。需要注意的是,Record 都是 final 的,且成员变量也都是 final 的,但是可以支持实现接口,例如,RunnableSerializable。因此,简单的方法参数封装或者 DTO 都是 Record 适应的场景。

// 成员变量 name 和 url
public record WebsiteRecord(String name, String url) implements Serializable {
public WebsiteRecord {
if (!url.toLowerCase().startsWith("http")) {
throw new IllegalArgumentException("Invalid URL address:" + url);
}
System.out.println("New Website:" + name);
}
}

Sealed Class(JDK17)

我们都知道,带有 final 修饰符的类是不能继承的,那如果有些情况,我们还是希望能扩展类的功能,但是限制只有某些类能继承呢?

在 Java 15 和 16 中引入了 Sealed Class 的功能预览,在 Java 17 中作为标准功能提供。Sealed Class 只允许特定的类继承。

// 只允许 Bicycle, Car, Truck 继承
public abstract sealed class Vehicle permits Bicycle, Car, Truck {...} // Car 带有 final 修饰符,防止进一步扩展
public final class Car extends Vehicle {...} // Truck 带有 sealed 修饰符,只允许特定的类继承
public sealed class Truck extends Vehicle permits PickTruck, CyberTruck {...} // Bicycle 带有 non-sealed 修饰符,任何类都可以继续扩展
public non-sealed class Bicycle extends Vehicle {...} // 任意扩展 Bicycle 类
public class Motor extends Bicycle {...}

使用 Sealed Class 还有几点注意事项:

  • 被许可的子类,必须在编译时能被父类访问到。
  • 被许可的子类,必须直接继承 Sealed 父类。
  • 被许可的子类,必须带有 finalsealednon-sealed 三个修饰符之一。
  • 被许可的子类必须在同一个 Java 模块中。

小结

JDK 版本的升级会引入新的编程语言功能,能进一步提高开发者编写程序的效率,以前需要好几行才能完成的功能,现在一行就搞定了。而新的概念和模块推出,也可以优化软件架构,减少对第三方库的依赖。

性能提升

JDK 17 不只是提供了开发者关心的一些编程语言功能,在与旧版本的 JDK 相比,性能也得到了很大的提升。与 JDk 8 和 JDK 11 这两个 LTS 相比,性能的提升主要得益于 JVM 中引入的新功能和改进,特别是 GC 方面的提升。

JDK 17 目前支持的垃圾收集器有:

  1. Parallel GC - JDK 8 和早期 JDK 的默认收集器,关注吞吐量,尝试在最小延迟的情况下尽快完成工作并提高吞吐量。
  2. Garbage First(G1)- JDK 9 开始使用的默认收集器,G1 关注总体平衡的性能,会尝试在吞吐量和延迟和吞吐量之间做平衡。
  3. ZGC(JDK 15) 和 Shenandoah GC(JDK 12)- 这两个收集器关注延迟,通过牺牲吞吐量达到低延迟。ZGC 的启用方式:-XX:+UseZGC
  4. Serial GC - 关注资源占用和启动时间。这个收集器更像是一个简化版的 Parallel GC,仅使用单线程进行处理。

可以看到,不同的收集器关注的性能方面是不一样的,而决定使用哪个收集器,有时候也不是非常容易。首先我们需要搞清楚的是,我们的性能目标是什么?希望提高业务的吞吐量、减少业务的延迟还是减少资源的占用?当然,我们希望最好这几点能同时满足,但是 GC 是没有办法在每个方面都做到极致的,毕竟设计这些 GC 的时候,是目标场景做了一些权衡的。再看一下这三大需要提升的场景:

  • 吞吐量 - 减少 GC 对完成单位时间内业务会话量的影响。
  • 延迟 - 减少 GC 对单个业务会话的影响。
  • 资源占用 - 减少 GC 使用的额外资源。

下面我们看看使用 16GB 堆内存和 SPECjbb 20151 基准测试对前三个收集器的测试结果。

吞吐量

测试结果中,数值越高表示性能越好,可以看到,JDK 17 中,ZGC 的性能提升了 20%,Parallel GC 和 G1 也分别有超过 10% 的提升。

延迟

从延迟的角度看,性能提升更加显著。在缩短 GC 暂停时间所投入的工作有了回报。其中 G1 提升效果最佳,ZGC 也不错。但是由于我们的基准测试是测量应用程序的延迟,所以还有数据在这个图表是看不到的。ZGC 在暂停时间上表现非常优异,下面的图表展示暂停时间的提升(数值越小越好),我们可以看到 ZGC 的超预期性能:

在这里,我们分析原始性能数字(每个柱子上的数字),因为图表按照数据尺度做了归一化,看上去有点奇怪。正如我们所看到的,JDK 17 中的 ZGC 的表现远低于其亚毫秒暂停时间的目标。G1 的目标是在延迟和吞吐量之间保持平衡,其暂停时间目标也低于其默认暂停时间目标 200ms。此图表最右边的柱子展示不同的收集器如何处理可伸缩性。ZGC 的设计是暂停时间不随堆大小而增大,我们清楚地看到,当堆扩展到 128 GB 时就是这种情况。从暂停时间的角度来看,G1 比 Parallel 更好地处理较大的堆,因为 G1 也具有保持暂停时间目标的逻辑。

资源占用

图中比较了三个不同收集器对本地内存的使用峰值。从结果看 Parallel 和 ZGC 都非常稳定,因此我们只能比较使用内存的绝对数值,而 G1 在这方面确实有所改进。

小结

无论使用哪种收集器,与旧版本相比,JDK 17 的整体性能都有很大的提升。在 JDK 8 中,默认使用 Parallel,但在 JDK 9 中改为了 G1。之后,G1 的改进速度就超过了 Parallel,但在有些情况下可能 Parallel 仍然是最好的选择。而 ZGC(JDK 15)的加入,为我们提供了第三种高性能 GC 的选择。

结语

开发环境方面,JDK 升级已经是大趋势,Tomcat 10.1 已经支持 Jarkata EE,Hibernate ORM 6.1 也要求 JDK 11+。而最著名的开源框架 Spring 已经带头走在了更新 JDK 的康庄大道上,那么其他使用 Spring 的技术框架肯定会慢慢跟上。比如,Jmix 也会在明年一季度的版本中适配 Spring 6 和 Boot 3;JHipster 更加激进,从 Spring RC 版本推出时,已经开始尝试适配。

运行环境方面,JDK 17+ 带来的性能提升总体来说能达到约 20%,但是对于实际运行的生产系统,这点提升可能不足以吸引运维部门做升级,毕竟堆硬件也能提升性能。而升级 JDK 还有可能引入新的安全问题。

因此,对开发人员来说,我们建议尽早切换至 JDK 17,尝试新的语言特性,提升自己的技能。尽管可能会面临新 JDK 中的安全问题,但是由于新版本正处在积极开发支持中,如果有问题也会很快解决。而对于企业来说,如果担心安全问题影响线上系统,可以缓一缓再使用。另外,如果考虑去 Oracle,则可以选择 OpenJDK,一些有实力的大企业也会使用 OpenJDK 构建自己的 JDK。

随着技术的发展,越来越多的框架的新版本都会陆续支持新版的 JDK,企业肯定会慢慢的融入升级的大军之中。

参考文章:

  1. https://spring.io/blog/2022/11/16/spring-framework-6-0-goes-ga
  2. https://reflectoring.io/java-release-notes/
  3. https://kstefanj.github.io/2021/11/24/gc-progress-8-17.html

是时候考虑升级 JDK 17 了的更多相关文章

  1. Centos6.7安装Pycharm及升级JDK

    首先到pycharm官网下载pycharm压缩包 wget https://www.jetbrains.com/pycharm/download/download-thanks.html?platfo ...

  2. ubuntun16.04不支持intel的最新网卡,升级到17.10后解决

    新买的神舟战神电脑.装了ubuntu16.04版本.但是安装后无线网卡无法使用无线网卡型号:是intel的一款网卡02:00.0 Network controller [0280]: Intel Co ...

  3. 整理eclipse,升级jdk环境小记录

    这2天在整理项目: 需要把eclipse 32位,jdk1.6 32位的更改为eclipse 64位,jdk1.8 64位版本的,于是我就在一台window7的电脑上直接操作,遇到了一下几点问题,记录 ...

  4. Linux 下升级JDK 1.7到1.8

    1.下载1.8的jdk rpm文件到linux系统 2.执行rpm -ivh jdk-8u151-linux-x64.rpm 选项详解: -a:查询所有套件: -b<完成阶段><套件 ...

  5. codeblocks升级c++17版本

    用了大半年的codeblocks,今天居然发现我还不会配置MINGW版本,现在C++已经更新到c++20了,而我还在用c++11,所以今天记录一下怎么更新c++版本吧. 其实步骤没有我们想象的那么困难 ...

  6. Log4j2又爆雷!2.16.0存在DOS风险,升级2.17.0可解决

    本以为,经过上周的2.16.0版本升级,Log4j2的漏洞修复工作,大家基本都要告一段落了. 万万没想到,就在周末,Log4j官方又发布了新版本:2.17.0 该版本主要修复安全漏洞:CVE-2021 ...

  7. CentOS升级JDK

    1.下载相应版本JDK: wget --no-check-certificate --no-cookie --header "Cookie: oraclelicense=accept-sec ...

  8. jenkins 升级jdk到1.8.0 报java.io.IOException:Unable to read /var/lib/jenkins/config.xml

    今天手动下载安装了jdk1.8.0, 并修改了配置文件,当前默认使用该版本的jdk.但是报出一下错误: 问题查到: https://issues.jenkins-ci.org/browse/JENKI ...

  9. centos6.x升级glibc-2.17

    glibc glibc是GNU发布的libc库,即c运行库.glibc是linux系统中最底层的api,几乎其它任何运行库都会依赖于glibc:它本身也提供了许多其它一些必要功能服务的实现: libc ...

  10. 升级jdk注意事项

    最好使用和编译的jdk相同版本的jre去执行.class程序. 今天在本地模拟部署项目到tomcat6就遇到了个坑. 我们项目使用的jdk1.7开发的,编译打war包放到webapps后,启动报错Ex ...

随机推荐

  1. ELK接收paloalto防火墙威胁日志并定位城市展示

    ELK接收paloalto防火墙威胁日志并定位城市展示 一.准备环境: 搭建好的ELK环境 palo alto防火墙(企业用的) 二.安装logstash并做好过滤 将palo alto日志打到一台c ...

  2. MySQL建表语句生成Golang代码

    1. 背景 对于后台开发新的需求时,一般会先进行各种表的设计,写各个表的建表语句 然后根据建立的表,写对应的model代码.基础的增删改查代码(基础的增删改查服务可以划入DAO(Data Access ...

  3. 输入法词库解析(三)紫光拼音词库.uwl

    详细代码:https://github.com/cxcn/dtool 前言 .uwl 是紫光拼音输入法(现在叫华宇拼音输入法)使用的词库. 解析 紫光的词库有点复杂,拼音用的索引,但是拼音表没有写在词 ...

  4. 【前端】在浏览器控制台,直接发Ajax请求

    我们在日常的开发的过程中,经常需要前端测试发送请求测试一些数据.但是由于一些session,cookie的存在,我们无法在postman上创建一些会话.那么这样,我们就可以在浏览器上直接发送Ajax请 ...

  5. uniapp scroll-view组件隐藏滚动条

    在pages.json增加如下配置 "globalStyle": { "navigationBarTextStyle": "black", ...

  6. portainer 1.24.2 升级到 portainer-ce 最新版

    官方升级步骤文档: https://docs.portainer.io/v/ce-2.9/start/upgrade/docker 若是现在的版本是 portainer-ce 2.0.0 ,看Opti ...

  7. 使用Kuboard界面在k8s上部署SpringCloud项目

    先安装Ingress Controller 安装Ingress Controller后,其他服务设置Ingress后就可以通过设置的域名进行访问了,就不用通过代理的方式或者ip:port的方式进行访问 ...

  8. Elasticsearch准实时索引实现(数据写入到es分片并存储到文件中的过程)

    溢写到文件系统缓存 当数据写入到ES分片时,会首先写入到内存中,然后通过内存的buffer生成一个segment,并刷到文件系统缓存中,数据可以被检索(注意不是直接刷到磁盘) ES中默认1秒,refr ...

  9. .NET 反向代理 YARP 自定义配置提供程序(Configuration Providers)

    介绍 基本 Yarp 示例显示从 appsettings.json 加载的代理配置.相反,代理配置可以从您选择的源以编程方式加载.您可以通过提供几个实现 IProxyConfigProvider 和 ...

  10. C语言指针笔记01

    int num = 90; 定义一个整型变量num int* ptr = &num; 定义一个整型指针变量ptr,指针变量ptr的类型取决于他所需要指向的变量,如这里,ptr要指向int类型变 ...