JDK 13新特性介绍1.1 JDK 各版本主要特性回顾JDK Version 1.0

1996-01-23 Oak(橡树)

JDK Version 1.1

1997-02-19

JDK Version 1.2JDK Version 1.3

2000-05-08 Kestrel(红隼)

JDK Version 1.4

2004-02-06 Merlin(隼)

JAVA 5

2004-09-30 Tiger(老虎)

JAVA 6

2006-12-11 Mustang(野马)

JAVA 7

2011-07-28 Dolphin(海豚)

JAVA 8

2014-03-18

JAVA 9

2017-09-22

JAVA 10

2018-03-21

根据官网的公开资料,共有12个重要特性,如下:

JAVA 11

2018-09-25

翻译后的新特性有:

JAVA 12

2018-09-25

翻译后的新特性有:

1.2 JDK 各版本支持周期

jdk版本怎么选择(一篇文带你了解JDK)(1)

为了更快地迭代,Java的更新从传统的以特性驱动的发布周期,转变为以时间驱动的(6 个月为周期)发布模式 每半年发布一个大版本,每个季度发布一个中间特性版本,并且承诺不会食言。通过这样的方式,开发团队可以把 些关键特性尽早合并到 JDK 之中,以快速得到开发者反馈,按照官方的说法,新的发布周期会严格遵循时间点,将于每年的3月份和9月份发布。所以 Java 11 的版本号是18.9(LTS,long term support)。Oracle 直到2023年9月都会为 Java 11 提供技术支持,而补丁和安全警告等扩展支持将持续到2026年。新的长期支持版本每三年发布一次,根据后续的发布计划,下一个长期支持版 Java 17 将于2021年发布

1.3 目前企业JDK版本使用现状

在 JDK 版本的世界里,从来都是 Oracle 发他的新版本,我们继续用我们的老版本。三年之前用 JDK 7,后来终于升级到了 JDK 8。自从升级了没多久,JDK 就开始了半年发一个新版本的节奏,陆续发布了 9 、10、11、12,直到(2019年9月17日)发布了 JDK13。

开发人员在生产中为其应用程序使用了哪些产品?我们可以看到Oracle JDK和OpenJDK在其他所有人的主导地位。2018年12月,由 Snyk 和 The Java Magazine 联合推出发布的 2018 JVM 生态调查报告 显示有 70% 的用户使用 Oracle JDK,21% 的用户使用 OpenJDK。

jdk版本怎么选择(一篇文带你了解JDK)(2)

Java 9中的JDK进行了重大的结构更改,许多人预测这将影响迁移和采用。从结果中我们可以看到(请注意,调查是在Java 10和Java 11发行之间进行的),Java 8仍然是Java的最主要版本-十分之八的受访者表示,他们的主要应用程序在生产中使用了Java 8。同样重要的是,使用更新版本的非Java 8受访者不到一半。2018 JVM 生态调查报告](https://snyk.io/blog/jvm-ecosystem-report-2018),[其中 Java 8 的使用者占到了 79%。

jdk版本怎么选择(一篇文带你了解JDK)(3)

1.4 JDK 13详细概述

​ 2019年9月17日,国际知名的OpenJDK开源社区发布了Java编程语言环境的最新版本OpenJDK 13,此次更新是继半年前 Java 12 这大版本发布之后的一次常规版本更新,在这一版中,主要带来了 ZGC 增强、更新 Socket 实现、Switch 表达式,文本块更新等方面的改动、增强。

1.5 JDK 13 新特性更新列表介绍

网站:http://openjdk.java.net/projects/jdk/13/

jdk版本怎么选择(一篇文带你了解JDK)(4)

1.6 启动IDEA创建模块集成JDK 13JDK 13新特性详解2.1 JEP 354 switch表达式(预览)引入

扩展switch分支选择语句的写法。Switch表达式在经过JDK 12的预览之后,在JDK 13中可以继续使用。

设计初衷

Java的switch语句是一个变化较大的语法(可能是因为Java的switch语句一直不够强大、熟悉swift或者js语言的同学可与swift的switch语句对比一下,就会发现Java的switch相对较弱),因为Java的很多版本都在不断地改进switch语句:JDK 12扩展了switch语句,使其可以用作语句或者表达式,并且传统的和扩展的简化版switch都可以使用。

JDK 12对于switch的增强主要在于简化书写形式,提升功能点。 下面简单回顾一下switch的进化阶段:

以前的switch程序

代码如下:

public class Demo01{ public static void main(String[] args){ // 声明变量score,并为其赋值为'C' var score = 'C'; // 执行switch分支语句 switch (score) { case 'A': System.out.println("优秀"); break; case 'B': System.out.println("良好"); break; case 'C': System.out.println("中"); break; case 'D': System.out.println("及格"); break; case 'E': System.out.println("不及格"); break; default: System.out.println("数据非法!"); } } }

这是经典的Java 11以前的switch写法 ,这里不能忘记写break,否则switch就会贯穿、导致程序出现错误(JDK 11会提示警告)。

JDK 13不需要break了

在JDK 12之前如果switch忘记写break将导致贯穿,在JDK 12对switch的这一贯穿性做了改进。你只要将case后面的冒号(:)改成箭头,那么你即使不写break也不会贯穿了,因此上面程序可改写如下形式:

public class Demo02{ public static void main(String[] args){ // 声明变量score,并为其赋值为'C' var score = 'C'; // 执行switch分支语句 switch (score){ case 'A' -> System.out.println("优秀"); case 'B' -> System.out.println("良好"); case 'C' -> System.out.println("中"); case 'D' -> System.out.println("及格"); case 'E' -> System.out.println("不及格"); default -> System.out.println("成绩数据非法!"); } } }

上面代码简洁很多了。

JDK 13的switch表达式

JDK 12之后的switch甚至可作为表达式了——不再是单独的语句。例如如下程序。

public class Demo03 { public static void main(String[] args) { // 声明变量score,并为其赋值为'C' var score = 'C'; // 执行switch分支语句 String s = switch (score) { case 'A' -> "优秀"; case 'B' -> "良好"; case 'C' -> "中"; case 'D' -> "及格"; case 'F' -> "不及格"; default -> "成绩输入错误"; }; System.out.println(s); } }

上面程序直接将switch表达式的值赋值给s变量,这样switch不再是一个语句,而是一个表达式.

JDK 13中switch的多值匹配

当你把switch中的case后的冒号改为箭头之后,此时switch就不会贯穿了,但在某些情况下,程序本来就希望贯穿比如我就希望两个case共用一个执行体!JDK 12之后的switch中的case也支持多值匹配,这样程序就变得更加简洁了。例如如下程序。

public class Demo04 { public static void main(String[] args) { // 声明变量score,并为其赋值为'C' var score = 'B'; // 执行switch分支语句 String s = switch (score) { case 'A', 'B' -> "上等"; case 'C' -> "中等"; case 'D', 'E' -> "下等"; default -> "成绩数据输入非法!"; }; System.out.println(s); } }

JDK 13的yielding a value

当使用箭头标签时,箭头标签右边可以是表达式、throw语句或是代码块。如果是代码块,需要使用yield语句来返回值。下面代码中的print方法中的default语句的右边是一个代码块。在代码块中使用yield来返回值。,JDK 13引入了一个新的yield语句来产生一个值,该值成为封闭的switch表达式的值。

public void print(int days) { // 声明变量score,并为其赋值为'C' var score = 'B'; String result = switch (score) { case 'A', 'B' -> "上等"; case 'C' -> "中等"; case 'D', 'E' -> "下等"; default -> { if (score > 100) { yield "数据不能超过100"; } else { yield score "此分数低于0分"; } } }; System.out.println(result); }

​ 在switch表达式中不能使用break。switch表达式的每个标签都必须产生一个值,或者抛出异常。switch表达式必须穷尽所有可能的值。这意味着通常需要一个default语句。一个例外是枚举类型。如果穷尽了枚举类型的所有可能值,则不需要使用default。在这种情况下,编译器会自动生成一个default语句。这是因为枚举类型中的枚举值可能发生变化。比如,枚举类型Color 中原来只有3个值:RED、GREEN和BLUE。使用该枚举类型的switch表达式穷尽了3种情况并完成编译。之后Color中增加了一个新的值YELLOW,当用这个新的值调用之前的代码时,由于不能匹配已有的值,编译器产生的default会被调用,告知枚举类型发生改变

小结

从以上案例可以看出JDK 12到JDK 13对switch的功能做了很大的改进,代码也十分的简化,目前来看switch依然是不支持区间匹配的,未来是否可以支持,我们拭目以待。

2.2 JEP 355 文本块升级(预览)引入

在Java中,在字符串文字中嵌入HTML,XML,SQL或JSON片段"..."通常需要先进行转义和串联的大量编辑,然后才能编译包含该片段的代码。该代码段通常难以阅读且难以维护,因此,如果具有一种语言学机制,可以比多行文字更直观地表示字符串,而且可以跨越多行,而且不会出现转义的视觉混乱,那么这将提高广泛Java类程序的可读性和可写性。从JDK 13到JDK 13开始文本块新特性的提出,提高了Java程序书写大段字符串文本的可读性和方便性。

HTML示例

使用“一维”字符串文字*

String html = "<html>\n" " <body>\n" " <p>Hello, world</p>\n" " </body>\n" "</html>\n";

使用“二维”文本块

String html = """ <html> <body> <p>Hello, world</p> </body> </html> """; System.out.println(""" Hello, itheima text blocks! """);

文本块是Java语言的新语法,可以用来表示任何字符串,具有更高的表达能力和更少的复杂度。文本块的开头定界符是由三个双引号 """ 开始,从新的一行开始字符串的内容。这里的新起的这行不属于字符串,只表示内容开始,是语法的一部分。以 """ 结束。 """ 可以紧跟字符串内容,也可以另起一行。另起一行时,字符串内容最后会留有新的一行。

""" line 1 line 2 line 3 """

等效于字符串文字:

"line 1\nline 2\nline 3\n"

或字符串文字的串联:

"line 1\n" "line 2\n" "line 3\n"

如果在字符串的末尾不需要行终止符,则可以将结束定界符放在内容的最后一行。例如,文本块:

""" line 1 line 2 line 3"""

等效于字符串文字:

"line 1\nline 2\nline 3"

文本块可以表示空字符串,尽管不建议这样做,因为它需要两行源代码:

String empty = """ """;

String a = """"""; // no line terminator after opening delimiter String b = """ """; // no line terminator after opening delimiter String c = """ "; // no closing delimiter (text block continues to EOF) String d = """ abc \ def """; // unescaped backslash (see below for escape processing)

HTML

使用原始字符串语法:

String html = "<html>\n" " <body>\n" " <p>Hello, world</p>\n" " </body>\n" "</html>\n";

使用文本块文本块语法:

String html = """ <html> <body> <p>Hello, world</p> </body> </html> """;

SQL

使用原始的字符串语法:

String query = "SELECT `EMP_ID`, `LAST_NAME` FROM `EMPLOYEE_TB`\n" "WHERE `CITY` = 'INDIANAPOLIS'\n" "ORDER BY `EMP_ID`, `LAST_NAME`;\n";

使用文本块语法:

String query = """ SELECT `EMP_ID`, `LAST_NAME` FROM `EMPLOYEE_TB` WHERE `CITY` = 'INDIANAPOLIS' ORDER BY `EMP_ID`, `LAST_NAME`; """;

多语言示例

使用原始的字符串语法:

ScriptEngine engine = new ScriptEngineManager().getEngineByName("js"); Object obj = engine.eval("function hello() {\n" " print('\"Hello, world\"');\n" "}\n" "\n" "hello();\n");

使用文本块语法:

ScriptEngine engine = new ScriptEngineManager().getEngineByName("js"); Object obj = engine.eval(""" function hello() { print('"Hello, world"'); } hello(); """);

缩进

java编译器会自动删除不需要的缩进:

System.out.println(""" Hello, itheima text blocks! """); // 结果 // > Hello, // > itheima // > text blocks! // > System.out.println(""" Hello, itheima text blocks! """); // 结果 // > Hello, // > itheima // > text blocks! // >

System.out.println(""" Hello, multiline text blocks! """); // 结果 // > Hello, // > multiline // > text blocks!

2.3 JEP 350 动态类数据共享归档

作用:允许在Java应用程序执行结束时动态归档类。归档的类将包括默认基层CDS归档中不存在的所有已加载应用程序类和库类。

CDS,是java 12的特性了,可以让不同 Java 进程之间共享一份类元数据,减少内存占用,它还能加快应用的启动速度。而JDK13的这个特性支持在Java application执行之后进行动态archive。存档类将包括默认的基础层CDS存档中不存在的所有已加载的应用程序和库类。也就是说,在Java 13中再使用AppCDS的时候,就不再需要这么复杂了。该提案处于目标阶段,旨在提高AppCDS的可用性,并消除用户进行试运行以创建每个应用程序的类列表的需要。

目标

JDK13这次对CDS增强的主要目的

意义

在JDK13中做的增强,可以只开启命令行选项完成上面过程,在程序运行的时候,动态评估那些类需要归档,同时支持内置的类加载器和用户定义的类加载器。

在第一次程序执行完成之后,会自动的将类进行归档,后续在启动项目的时候也无需指定要使用哪些归档,整个过程看起来更加的透明。

2.4 JEP 351 ZGC 增强: ZGC 释放未使用内存目标

作用:将未使用的堆内存还给系统(即未提交的内存空间)

详解

ZGC 是 Java 11 中引入的最为瞩目的垃圾回收特性,是一种可伸缩、低延迟的垃圾收集器,不过在 Java 11 中是实验性的引入,主要用来改善 GC 停顿时间,并支持几百 MB 至几个 TB 级别大小的堆,并且应用吞吐能力下降不会超过 15%,目前只支持 Linux/x64 位平台的这样一种新型垃圾收集器。

通过在实际中的使用,发现 ZGC 收集器中并没有像 Hotspot 中的 G1 和 Shenandoah 垃圾收集器一样,能够主动将未使用的内存释放给操作系统的功能。对于大多数应用程序来说,CPU 和内存都属于有限的紧缺资源,特别是现在使用的云上或者虚拟化环境中。如果应用程序中的内存长期处于空闲状态,并且还不能释放给操作系统,这样会导致其他需要内存的应用无法分配到需要的内存,而这边应用分配的内存还处于空闲状态,处于"忙的太忙,闲的太闲"的非公平状态,并且也容易导致基于虚拟化的环境中,因为这些实际并未使用的资源而多付费的情况。由此可见,将未使用内存释放给系统主内存是一项非常有用且亟需的功能。

Java 13 中对 ZGC 的改进,主要体现在下面几点:

Java 13 中,ZGC 内存释放功能,默认情况下是开启的,不过可以使用参数:-XX:-ZUncommit 显式关闭。

2.5 JEP 353 重新实现旧版Socket API引入

现在已有的 java.net.Socket 和 java.net.ServerSocket 以及它们的实现类,都可以回溯到 JDK 1.0 时代了。原始socket的维护和调试都很痛苦。实现类还使用了线程栈作为 I/O 的缓冲,导致在某些情况下还需要增加线程栈的大小。该实现还存在几个并发问题,需要彻底解决。在未来的网络世界,要快速响应,不能阻塞本地方法线程,当前的实现不适合使用了。

详解

新的实现类

在 Java 13 之前,通过使用 PlainSocketImpl 作为 SocketImpl 的具体实现。

Java 13 中的新底层实现,引入 NioSocketImpl 的实现用以替换 SocketImpl 的 PlainSocketImpl 实现,此实现与 NIO(新 I/O)实现共享相同的内部基础结构,并且与现有的缓冲区高速缓存机制集成在一起,因此不需要使用线程堆栈。除了这些更改之外,还有其他一些更便利的更改,如使用 java.lang.ref.Cleaner 机制来关闭套接字(如果 SocketImpl 实现在尚未关闭的套接字上被进行了垃圾收集),以及在轮询时套接字处于非阻塞模式时处理超时操作等方面。

全新实现的 NioSocketImpl 来替换JDK1.0的PlainSocketImpl。

代码说明

运行一个实例化Socket和ServerSocket的类将显示这个调试输出。这是默认的(新的)。

Module java.base Package java.net Class SocketImpl public abstract class SocketImpl implements SocketOptions { private static final boolean USE_PLAINSOCKETIMPL = usePlainSocketImpl(); private static boolean usePlainSocketImpl() { PrivilegedAction<String> pa = () -> NetProperties.get("jdk.net.usePlainSocketImpl"); String s = AccessController.doPrivileged(pa); return (s != null) && !s.equalsIgnoreCase("false"); } /** Creates an instance of platform's SocketImpl */ @SuppressWarnings("unchecked")static <S extends SocketImpl & PlatformSocketImpl> S createPlatformSocketImpl(boolean server) { if (USE_PLAINSOCKETIMPL) { return (S) new PlainSocketImpl(server); } else { return (S) new NioSocketImpl(server); } } }

SocketImpl的USE_PLAINSOCKETIMPL取决于usePlainSocketImpl方法,而它会从NetProperties读取 dk.net.usePlainSocketImpl配置,如果不为null且不为false,则usePlainSocketImpl方法返回true; createPlatformSocketImpl会根据USE_PLAINSOCKETIMPL来创建PlainSocketImpl或者NioSocketImpl。

通过这些更改,Java Socket API 将更易于维护,更好地维护将使套接字代码的可靠性得到改善。同时 NIO 实现也可以在基础层面完成,从而保持 Socket 和 ServerSocket 类层面上的不变。

2.6 其他特性

上面列出的是大方面的特性,除此之外还有一些api的更新及废弃,主要见 JDK 13 Release Notes,这里举几个例 子。 https://jdk.java.net/13/release-notes

增加项

添加FileSystems.newFileSystem(Path, Map<String, ?>) Method 新的java.nio.ByteBuffer Bulk get/put Methods Transfer Bytes Without Regard to Buffer Position 支持Unicode 12.1 添加-XX:SoftMaxHeapSize Flag,目前仅仅对ZGC起作用 ZGC的最大heap大小增大到16TB

移除项

移除awt.toolkit System Property 移除Runtime Trace Methods 移除-XX: AggressiveOpts 移除Two Comodo Root CA Certificates、Two DocuSign Root CA Certificates 移除内部的com.sun.net.ssl包

废弃项

废弃-Xverify:none及-noverify 废弃rmic Tool并准备移除 废弃javax.security.cert并准备移除

已知问题

不再支持Windows 2019 Core Server 使用ZIP File System (zipfs) Provider来更新包含Uncompressed Entries的ZIP或JAR可能造成文件损坏

展望

Java 在更新发布周期为每半年发布一次之后,在合并关键特性、快速得到开发者反馈等方面,做得越来越好。从 Java 11 到 Java 13,目前确实是严格保持半年更新的节奏。Java 13 版本的发布带来了些新特性和功能增强、性能提升和改进尝试。

以上包含的5个特性,能够改变开发者的编码风格的主要有Text Blocks和Switch Expressions两个新特性,但是这两个特性还处于预览阶段。

而且,JDK13并不是LTS(长期支持)版本,如果你正在使用Java 8(LTS)或者Java 11(LTS),暂时可以不必升级到Java 13。

最后,祝大家早日学有所成,拿到满意offer,快速升职加薪,走上人生巅峰。

可以的话请给我一个三连支持一下我哟

,