blog/source/_posts/nf-java14.md
2021-05-24 21:01:20 +08:00

189 lines
7.1 KiB
Markdown
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

---
title: Java 14的新特性
tags:
- JVM
- Java
- Java 14
- 新特性
- switch
- 文本块
- 打包
- Package
- Record Class
categories:
- - JVM
- Java
keywords: 'Java,Java 14,新特性,switch,文本,文本块,打包,内存,记录类'
date: 2021-05-24 11:16:10
---
Java 14带来了许多新功能尤其是把之前一直处于预览状态的功能进行了实装。而且新加入的一些正在孵化的功能也是大大增强了Java的适用面。<!-- more -->
本系列的文章有:
- {% post_link nf-java8 %}
- {% post_link nf-java9 %}
- {% post_link nf-java10 %}
- {% post_link nf-java11 %}
- {% post_link nf-java12 %}
- {% post_link nf-java13 %}
- {% post_link nf-java14 %}
- {% post_link nf-java15 %}
- {% post_link nf-java16 %}
## Switch表达式
Switch表达式是在Java 12中以预览版的身份被添加进来的。在Java语言的历史中`switch`关键字始终是作为一个语句出现的。例如我们经常会用到的星期枚举类。
```java
public enum WeekDay {
MONDAY, TUESDAY, WEDNESDAY, THUSDAY, FRIDAY, SATURDAY, SUNDAY;
public bool isHoliday() {
boolean isTodayHoliday = false;
swtich (this) {
case MONDAY:
case TUESDAY:
case WEDNESDAY:
case THUSDAY:
case FRIDAY:
isTodayHoliday = false;
break;
case SATURDAY:
case SUNDAY:
isTodayHoliday = true;
break;
default:
throw new IllegalArgumentException("Illegal days");
}
return isTodayHoliday;
}
}
```
在这个示例中,用到了一个中间缓存变量来保存判断结果,而且整个`switch`结构还使用了fall-through的特性。这些特性集中在一起就十分容易让开发人员犯各种错误。新版的`switch`结构被改成了一个表达式而且还没有了fall-through带来的缺点语法整体也被精简了许多。
下面这个示例是使用`switch`表达式语法重写上面这个示例。
```java
public enum WeekDay {
MONDAY, TUESDAY, WEDNESDAY, THUSDAY, FRIDAY, SATURDAY, SUNDAY;
public bool isHoliday() {
return switch (this) {
case MONDAY, TUESDAY, WEDNESDAY, THURSDAY, FRIDAY -> false;
case SATURDAY, SUNDAY -> true;
default -> throw new IllegalArgumentException("Illegal days");
};
}
}
```
## 改进的NPE提示
例如有以下会产生`NullPointerException`的语句。
```java
int[] arr = null;
arr[0] = 1;
```
在Java 14之前的版本中会得到以下的错误提示。
```
Exception in thread "main" java.lang.NullPointerException
at xyz.archgrid.demo.App.main(App.java:27)
```
但是在Java 14中NPE的错误提示被加强了而且变得更加有可读性和目的性例如上面示例中的提示就会变成下面这样。
```
java.lang.NullPointerException: Cannot store to int array because "arr" is null
```
## 预览版功能
## 文本块
文本块自Java 13以来还依旧处于预览状态。但是文本块功能在Java 14中增加了两个转义字符。
- `\`加入到文本块中,可以表示两行之间的连接,只是用于文本块中的文本产生阅读视觉上的换行,但是其中并不真正的换行。
- `\s`表示一个空格。
!!! note ""
虚拟换行`\`仅仅是防止一行字符串写的太长,但是如果直接使用换行又会导致字符串的实际换行存在的。
### instanceOf中的模式匹配
在Java 14中移除了`instanceOf`的一些样板代码,例如在之前的版本中,如果变量的类型匹配上了,那么接下来就一定会产生如下的操作。
```java
if (obj instanceof String) {
String str = (String) obj;
// 完成其他对于字符串变量str的操作
}
```
为了移除这个并不必要的字符串变量声明Java 14把变量声明与`instanceOf`操作符合并了。于是就变成了下面这个样子。
```java
if (obj instanceOf String str) {
// 完成其他对于字符串变量str的操作。
}
```
### 记录类
在Java中定义一个POJO还是比较繁琐的除了要定义用于保存数据的字段以外还需要定义相应的getter和setter必要的时候还需要重载`hashCode()`和`equals()`方法。所以这也是Lombok库能够引起大部分开发者共鸣的原因。
比如借助Lombok库我们可以这样来定义一个数据类。
```java
@Data
public class User {
private int uid;
private String username;
private String password;
}
```
为了简化POJO的书写也是从其他语言中借鉴Java 14引入了一个名为记录类的预览版功能。这是一个新的关键字`record`它用来声明一个特殊的类效果与使用Lombok的`@Data`修饰的普通类几乎相同。但记录类的设计主旨是用于持有不可变数据的所以其在使用的时候还需要与以前的POJO区别开。
如果使用`record`关键字,那么上面这个示例就可以被改写成下面这个样子。
```java
public record User(int uid, String username, String password) { }
```
整个POJO类的声明就被简化成了类似于构造函数的样子那么现在这个类就可以如同以下示例中这样来使用了。
```java
User user = new User(0, "John", "123");
assertEquals(0, user.id());
assertEquals("John", user.username());
```
可以看出,记录类中的字段都被转换成了方法,可以通过方法的调用来获取字段的值。而且记录类会自动提供`hashCode()`、`equals()`和`toString()`方法。
## 孵化功能
孵化功能与预览功能不同,这些功能比预览功能更加的实验性。孵化功能一般都被放置于`jdk.incubator`包中。
### 外部内存访问
外部内存访问API提供了程序对于JVM以外的系统内存的访问能力。这套API允许Java程序用十分安全、有效率的途径来访问外部内存。
新的外部内存访问API的功能主要是通过`MemorySegment`、`MemoryAddress`和`MemoryLayout`这几个类来提供的,可以允许程序同时访问堆内存和非堆内存。
### 打包工具
传统上Java程序所遵循的都是“Build once, run everywhere”的跨平台理念。在这个历年的支持下Java程序一般都是以JAR包的形式出现和发布在运行的时候都需要提供提供JAR运行所需的JVM环境。这也就是说用户至少需要在他们的机器上安装JRE。
但是无论是在Windows、macOS还是在Linux上很多程序都是以安装包的形式出现的用户只需要运行安装包就可以把程序安装到系统中并且可以直接拥有启动程序的快捷方式。
用户所期望的这些程序安装方式Java都不能提供。所以在Java 14中将一套专门的打包工具列入了孵化项目。
在这套工具中,常用的有两个工具:`jlink`和`jpackage`。其中`jlink`主要用于JDK和JRE的裁剪仅为程序保留程序运行所需的最小模块。`jpackage`主要用来对程序和裁剪过后的JRE进行打包形成可执行文件例如Windows中的`.exe`macOS中的`.app`),并同时形成相应的安装包。
打包工具的出现将极大的改变Java程序的发布和部署。