JDK13新特性,让开发效率更快一步
推荐阅读:
闭关修炼21天,“啃完”283页pdf,我终于4面拿下字节跳动offer
肺炎在家“闭关”,阿里竟发来视频面试,4面顺利拿下offer
JDK13到今天已经发布快2个月了,之前有零零散散的试过一些新的特性,但却没有整体的整理一下。想到作为Java开发,连使用的JDK(Java Developerment Kit)有什么特性都不清楚,实在是有些不应该,想要进阶为更有价值的JAVA开发人员,一定要跟得上JDK的最新特性。
那再来看下,这份迟来的JDK新特性一览
JDK13
所有的JDK特性都会在JEP进度中提出来和跟踪:openjdk.java.net/jeps/0
对于每个JDK版本的有什么特性在对应JDK主页中查看:openjdk.java.net/projects/jd…
JDK13主要有5个特性:
- 350: Dynamic CDS Archives 参考: www.jianshu.com/p/0b8a9d137…
- 351: ZGC: Uncommit Unused Memory 参考:www.jianshu.com/p/18fc5a042…
- 353: Reimplement the Legacy Socket API
- 354: Switch Expressions (Preview)
- 355: Text Blocks (Preview)
Dynamic CDS Archives
在JDK10中被引入的新特性,但是当时创建步骤比较繁琐。
# JDK10中需要的步骤1. 需要指定要归档那些类 -XX:DumpLoadedClassList=classes.lst2. 创建归档 -Xshare:dump -XX:SharedArchiveFile -XX:SharedClassListFile=classes.lst3. 使用归档 -Xshare:on -XX:SharedArchiveFile
在JDK13中引入新的选项,在程序退出时自动归档:
java -XX:ArchiveClassesAtExit=app.jsa -cp app.jar HelloDemo
使用归档步骤与之前相同,默认-Xshare:on是开启的
类加载过程:
加载->验证->准备->解析->初始化->使用->卸载
- 加载:找到Class的位置,从Class位置读取Class文件内容
- 验证:文件格式的验证、元数据的验证、字节码验证和符号引用验证。
- 准备:正式为类变量分配内存并设置类变量初始值的阶段,这些内存都将在方法区中分配。存储为JDK内部数据结构
- 解析:虚拟机将常量池中的符号引用转化为直接引用的过程,解析接口,字段解析
- 初始化:创建类
CDS的设计目的主要为了提升启动应用时的速度,class-data只需要创建一次,后续重复使用,减少了加载,验证,准备阶段。可能会有解析阶段
参考:App CDS实战
ZGC: Uncommit Unused Memory
ZGC从JDK11中被引入进来,在进行GC的时候保证更短的停顿时间,10ms以下,在JDK13中新增了归还未提交,未使用的内存给操作系统
ZGC由许多的ZPage组成,Zpage是不同大小的内存区域,分为小、中、大。当ZGC压缩内存时,Zpage被清空到ZPageCache中,ZpageCache是准备随时被用到的区域,如果被使用,会立刻从ZpageCache中移除到Zpage中,但是如果ZpageCache中的Zpage长时间未使用,则变为未提交使用的内存,后续可还给操作系统。
When ZGC compacts the heap, ZPages are freed up and inserted into a page cache, the ZPageCache.
#设置一个时间多久从ZpageCache中移除(evict)Zpage-XX:+UnlockExperimentalVMOptions -XX:+ZUncommit -XX:ZUncommitDelay=<seconds>
参考:ZGC完全指南
Reimplement the Legacy Socket API
JDK底层对Socket的实现非常的古老,从JDK1.0中被使用一直到现在,底层为很早的Java和C代码,对于开发JDK的人来说,非常的难以维护和Debug,因此重新实现了Socket API的接口。
- JDK13之前,使用PlainSocketImpl
- JDK13引入了,NioSocketImpl替换PlainSocketImpl。
来一个HelloWorld案例:
public class HelloApp { public static void main(String[] args) { try (ServerSocket serverSocket = new ServerSocket(8888)) { boolean running = true; System.out.println("listened 8888"); while (running) { Socket clientSocket = serverSocket.accept(); //do something with clientSocket } } catch (IOException e) { e.printStackTrace(); } }}
但是我们仍然可以切换为PlainSocketImpl。需配置jdk.net.usePlainSocketImpl
/Library/Java/JavaVirtualMachines/jdk-13.jdk/Contents/Home/bin/java -XX:+TraceClassLoading me/aihe/HelloApp.java | grep -i socketI
/Library/Java/JavaVirtualMachines/jdk-13.jdk/Contents/Home/bin/java -XX:+TraceClassLoading -Djdk.net.usePlainSocketImpl me/aihe/HelloApp.java | grep -i socketI
Switch Expressions (Preview)
引入了一个新的关键字yield用于返回switch语句的内容。最开始我们写switch语句都要在语句之前做一些初始化变量,现在可以直接得到swicth语句额返回结果
最开始的switch写法:
int numLetters;switch (day) { case MONDAY: case FRIDAY: case SUNDAY: numLetters = 6; break; case TUESDAY: numLetters = 7; break; case THURSDAY: case SATURDAY: numLetters = 8; break; case WEDNESDAY: numLetters = 9; break; default: throw new IllegalStateException("Wat: " + day);}
在JDK13中可以这样写:
# 没有逻辑的返回int numLetters = switch (day) { case MONDAY, FRIDAY, SUNDAY -> 6; case TUESDAY -> 7; case THURSDAY, SATURDAY -> 8; case WEDNESDAY -> 9;};# 逻辑较多的处理 String result = switch (number) { case 1, 2: // 逻辑代码 yield "one or two"; case 3: // 逻辑代码 yield "three"; case 4, 5, 6: yield "four or five or six"; default: yield "unknown"; }; return result;
Text Blocks (Preview)
最开始写长字符串的时候,往往要使用多个字符串拼接,一是浪费性能,而是看起来很难看。尤其写HTML字符串或者SQL语句时。
// 比如HTMLString html = "<html>\n" + " <body>\n" + " <p>Hello, world</p>\n" + " </body>\n" + "</html>\n";
现在可以写成:
String html = """ <html> <body> <p>Hello, world</p> </body> </html> """;
注意:
- 其中有个细微的区别,是开头"""之后必须另起一行,另外结尾的"""是否另起一行有不同的效果
- 注意在使用的时候每一行可能需要处理两边的空格
"""line 1line 2line 3"""=>"line 1\nline 2\nline 3\n"
"""line 1line 2line 3"""=>"line 1\nline 2\nline 3"
最后
JDK13在一定程度上还是可以加快我们的开发速度...,最重要的是其归档特性可以大大减少我们应用的启动时间,ZGC则让我们在内存吃紧时,又带来了福音。
值得一试!