背景
这么多新版本的 JDK,而且2022年还会推出 JDK 18 和 JDK 19,为什么 Spring 选择了 JDK 17呢。
JDK9新特性(2017年9月)
模块化
提供了List.of()、Set.of()、Map.of()和Map.ofEntries()等工厂方法
接口支持私有方法
Optional 类改进
多版本兼容Jar包
JShell工具
try-with-resources的改进
Stream API的改进
设置G1为JVM默认垃圾收集器
JDK10新特性(2018年3月)
局部变量类型推断,类似JS可以通过var来修饰局部变量,编译之后会推断出值的真实类型
不可变集合的改进
并行全垃圾回收器 G1,来优化G1的延迟
线程本地握手,允许在不执行全局VM安全点的情况下执行线程回调,可以停止单个线程,而不需要停止所有线程或不停止线程
Optional新增orElseThrow()方法
类数据共享
Unicode 语言标签扩展
JDK11新特性(2018年9月)(LTS版本)
增加一些字符串处理方法
用于 Lambda 参数的局部变量语法
Http Client重写,支持HTTP/1.1和HTTP/2 ,也支持 websockets
可运行单一Java源码文件,如:java Test.java
ZGC:可伸缩低延迟垃圾收集器,ZGC可以看做是G1之上更细粒度的内存管理策略。由于内存的不断分配回收会产生大量的内存碎片空间,因此需要整理策略防止内存空间碎片化,在整理期间需要将对于内存引用的线程逻辑暂停,这个过程被称为"Stop the world"。只有当整理完成后,线程逻辑才可以继续运行。(并行回收)
支持 TLS 1.3 协议
Flight Recorder(飞行记录器),基于OS、JVM和JDK的事件产生的数据收集框架
JDK12新特性(2019年3月)
Switch 表达式扩展,可以有返回值
新增NumberFormat对复杂数字的格式化
字符串支持transform、indent操作
新增方法Files.mismatch(Path, Path)
Teeing Collector
支持unicode 11
Shenandoah GC,新增的GC算法
JDK13新特性(2019年9月)
Switch 表达式扩展,switch表达式增加yield关键字用于返回结果,作用类似于return,如果没有返回结果则使用break
文本块升级 """ ,引入了文本块,可以使用"""三个双引号表示文本块,文本块内部就不需要使用换行的转义字符
SocketAPI 重构,Socket的底层实现优化,引入了NIO
FileSystems.newFileSystem新方法
JDK14新特性(2020年3月)
instanceof模式匹配,instanceof类型匹配语法简化,可以直接给对象赋值,如if(obj instanceof String str),如果obj是字符串类型则直接赋值给了str变量
引入Record类型,类似于Lombok 的@Data注解,可以向Lombok一样自动生成构造器、equals、getter等方法;
Switch 表达式-标准化
改进 NullPointerExceptions提示信息,打印具体哪个方法抛的空指针异常,避免同一行代码多个函数调用时无法判断具体是哪个函数抛异常的困扰,方便异常排查;
删除 CMS 垃圾回收器
JDK15新特性(2020年9月)
EdDSA 数字签名算法
Sealed Classes(封闭类,预览),通过sealed关键字修饰抽象类限定只允许指定的子类才可以实现或继承抽象类,避免抽象类被滥用
Hidden Classes(隐藏类)
移除 Nashorn JavaScript引擎
JDK16新特性(2021年3月)
允许在 JDK C ++源代码中使用 C ++ 14功能
ZGC性能优化,去掉ZGC线程堆栈处理从安全点到并发阶段
增加 Unix 域套接字通道
弹性元空间能力
JDK17新特性(2021年9月)(LTS版本)
Free Java License
JDK 17 将取代 JDK 11 成为下一个长期支持版本
Spring 6 和 Spring Boot 3需要JDK17
移除实验性的 AOT 和 JIT 编译器
恢复始终执行严格模式 (Always-Strict) 的浮点定义
正式引入密封类sealed class,限制抽象类的实现
重要特性详解
Java 模块化
一个模块通常只是一个 jar 文件,在根目录下有一个文件module-info.class。
module helloworld {
exports com.alibaba.eight;
}
module test {
requires helloworld;
}
本地变量类型推断
MyObject value = new MyObject();
var value = new MyObject();
HTTP客户端API-响应式流实现的HttpClient
完整支持HTTP 2.0 或者HTTP 1.1
支持 HTTPS/TLS
有简单的阻塞使用方法
支持异步发送,异步时间通知
支持WebSocket
支持响应式流
语法糖
示例:
List<Student> list = Arrays.asList(
new Student("唐一", 55),
new Student("唐二", 60),
new Student("唐三", 90));
//平均分 总分
String result = list.stream().collect(Collectors.teeing(
Collectors.averagingInt(Student::getScore),
Collectors.summingInt(Student::getScore),
(s1, s2) -> s1 + ":" + s2));
//最低分 最高分
String result2 = list.stream().collect(Collectors.teeing(
Collectors.minBy(Comparator.comparing(Student::getScore)),
Collectors.maxBy(Comparator.comparing(Student::getScore)),
(s1, s2) -> s1.orElseThrow() + ":" + s2.orElseThrow()
));
System.out.println(result);
System.out.println(result2);
List<String> list = Arrays.asList("1", "2", "3");
//之前这样写
List<Integer> oneList = list.stream()
.map(Integer::parseInt)
.collect(Collectors.toList());
//现在可以这样写
List<Integer> twoList = list.stream()
.map(Integer::parseInt)
.toList();
支持箭头表达式(jdk12预览 jdk14标准)
此更改扩展了switch 语句以便它可以用作语句或表达式。不必为break每个 case 块定义一个语句,我们可以简单地使用箭头语法。
boolean isWeekend = switch (day) {
case MONDAY, TUESDAY, WEDNESDAY, THURSDAY, FRIDAY -> false;
case SATURDAY, SUNDAY -> true;
default -> throw new IllegalStateException("Illegal day entry :: " + day);
};
int size = 3;
String cn = switch (size) {
case 1 -> "壹";
case 2 -> "贰";
case 3, 4 -> "叁";
default -> "未知";
};
System.out.println(cn);
//要使用此预览功能,我们必须在应用程序启动期间使用–enable-preview标志明确指示 JVM。
yield关键字(jdk13)
public class SwitchTest {
public static void main(String[] args) {
var me = 4;
var operation = "平方";
var result = switch (operation) {
case "加倍" -> {
yield me * 2;
}
case "平方" -> {
yield me * me;
}
default -> me;
};
System.out.println(result);
}
}
String json = "{\r\n" + "\"name\" : \"lingli\",\r\n" + "\"website\" : \"https://www.alibaba.com/\"\r\n" + "}";
现在让我们使用字符串文本块编写相同的 JSON :
String json = """
{
"name" : "Baeldung",
"website" : "https://www.alibaba.com/"
}
""";
之前:
Object obj = "大阳";
if (obj instanceof String) {
String t = (String) obj;
// TODO
}
现在:
Object obj = "大阳";
if (obj instanceof String t) {
// TODO 此时t已经是String类型了
}
/**
* record 记录类
* 你也可以覆写equals() hashCode() toString()方法,不用写get、set了
* @author DAYANG
*/
record User(String name, Integer age) {
public String toString() {
return "User[" +
"name='" + name + '\'' +
", age=" + age +
']';
}
public boolean equals(Object obj) {
return false;
}
public int hashCode() {
return 0;
}
}
JVM
其他
/**
* 定义一个抽象密封类Pet,它的实现类只能是Dog, Cat这两个,其他的实现类均不允许
*/
public abstract sealed class Pet
permits Dog, Cat {}
final class Dog extends Pet {
}
final class Cat extends Pet {
}
//密封的类和接口限制了其他类或接口可以扩展或实现它们
public sealed interface Shape{
final class Planet implements Shape {}
final class Star implements Shape {}
final class Comet implements Shape {}
}
public abstract sealed class Test{
final class A extends Test {}
final class B extends Test {}
final class C extends Test {}
}
总结和展望
总结
展望
[1]https://zhuanlan.zhihu.com/p/480293185
[2]https://blog.oxings.com/article/31.html
[3]https://blog.csdn.net/best_luxi/article/details/122543074