Java 11 于 2018 年 9 月 25 日正式发布,且为长期支持版本(Long-Term-Support - LTS)。
下面为 Oracle Java SE 产品的一些关键产品日期示例(Oracle Java SE Product Releases):
Java 11 对 Java 9 中引入并在 Java 10 中更新的 Http Client API 进行了标准化,实现了几乎完全重写,并且现在是完全异步非阻塞的 (以前的 HTTP/1.1 的实现是阻塞的)。
Http Client 的包名由 jdk.incubator.http
改为 java.net.http
。该 API 通过 CompleteableFutures 提供非阻塞请求和响应语义。
示例:
var request = HttpRequest.newBuilder()
.uri(URI.create("https://www.baidu.com/"))
.GET()
.build();
var client = HttpClient.newHttpClient();
// 同步
HttpResponse<String> response = client.send(request, HttpResponse.BodyHandlers.ofString());
System.out.println(response.body());
// 异步
client.sendAsync(request, HttpResponse.BodyHandlers.ofString())
.thenApply(HttpResponse::body)
.thenAccept(System.out::println);
JEP 321 HTTP Client (Standard)
Java 11 新增了一些 String 方法:
class strMethodsTest {
public static void main(String args[]) {
String str1 = "";
System.out.println(str1.isBlank());
String str2 = "pointerJava";
System.out.println(str2.isBlank());
}
}
输出结果:
true
false
import java.util.*;
class strMethodsTest {
public static void main(String args[]) {
String str = "pointer\njava";
System.out.println(str.lines().count());
System.out.println(str.lines().collect(Collectors.toList()));
}
}
输出结果:
2
pointer
java
class strMethodsTest {
public static void main(String args[]) {
String str = "pointerJava";
System.out.println(str.repeat(2));
}
}
输出结果:
pointerJavapointerJava
class strMethodsTest {
public static void main(String args[]) {
String str = " pointerJava";
System.out.println(str.stripLeading());
}
}
输出结果:
pointerJava
class strMethodsTest {
public static void main(String args[]) {
String str = "pointerJava ";
System.out.println(str.stripTrailing());
}
}
输出结果:
pointerJava
class strMethodsTest {
public static void main(String args[]) {
String str = " pointerJava ";
System.out.println(str.strip());
}
}
输出结果:
pointerJava
详细介绍可以参考美团技术团队创作的:《新一代垃圾回收器ZGC的探索与实践》
JEP 333: ZGC: A Scalable Low-Latency Garbage Collector (Experimental)
新增了 isEmpty() 方法来判断指定的 Optional 对象是否为空。
var op = Optional.empty();
System.out.println(op.isEmpty());
从 Java 10 开始,引入了局部变量类型推断(Local Variable Type Inference)功能,它可以让我们使用关键字 var
声明局部变量(而不是实际类型),而由编译器根据变量初始化的值自动推断出类型。
Java 10 中对 var
关键字存在几个限制
JDK 11 允许在 lambda 表达式中使用 var
进行参数声明:
public class VarInLambdaExample {
public static void main(String[] args) {
IntStream.of(1, 2, 3, 5, 6, 7)
.filter((var i) -> i % 2 == 0)
.forEach(System.out::println);
}
}
public class WithoutVarInLambdaExample {
public static void main(String[] args)
IntStream.of(1, 2, 3, 5, 6, 7)
.filter(i -> i % 2 == 0)
.forEach(System.out::println);
}
}
输出结果都为:
2
6
JEP 323: Local-Variable Syntax for Lambda Parameters
该特性不需要首先使用javac工具编译java源文件。 你可以使用 Java 命令直接运行该文件,并隐式编译。
例如:如果名为 HelloWorld.java 的文件包含一个名为 hello.World 的类,那么该命令:
$ java HelloWorld.java
等同于:
$ javac HelloWorld.java
$ java -cp hello.World
Epsilon 垃圾收集器不执行任何实际的垃圾回收操作(没有实际的内存回收机制),它只是简单地分配内存并忽略所有的垃圾对象。这对于一些特殊的性能测试和调优场景可能很有用。
使用场景:
性能测试 :什么都不执行的 GC 非常适合用于 GC 的差异性分析。no-op (无操作)GC 可以用于过滤掉 GC 诱发的性能损耗,比如 GC 线程的调度,GC 屏障的消耗,GC 周期的不合适触发,内存位置变化等。此外有些延迟者不是由于 GC 引起的,比如 scheduling hiccups, compiler transition hiccups,所以去除 GC 引发的延迟有助于统计这些延迟。
内存压力测试:在测试 Java 代码时,确定分配内存的阈值有助于设置内存压力常量值。这时 no-op 就很有用,它可以简单地接受一个分配的内存分配上限,当内存超限时就失败。例如:测试需要分配小于 1G 的内存,就使用-Xmx1g 参数来配置 no-op GC,然后当内存耗尽的时候就直接 crash。
VM 接口测试:以 VM 开发视角,有一个简单的 GC 实现,有助于理解 VM-GC 的最小接口实现。它也用于证明 VM-GC 接口的健全性。
极度短暂 job 任务:一个短声明周期的 job 任务可能会依赖快速退出来释放资源,这个时候接收 GC 周期来清理 heap 其实是在浪费时间,因为 heap 会在退出时清理。并且 GC 周期可能会占用一会时间,因为它依赖 heap 上的数据量。
延迟改进:对那些极端延迟敏感的应用,开发者十分清楚内存占用,或者是几乎没有垃圾回收的应用,此时耗时较长的 GC 周期将会是一件坏事。
吞吐改进:即便对那些无需内存分配的工作,选择一个 GC 意味着选择了一系列的 GC 屏障,所有的 OpenJDK GC 都是分代的,所以他们至少会有一个写屏障。避免这些屏障可以带来一点点的吞吐量提升。
JEP 318: Epsilon: A No-Op Garbage Collector (Experimental)
JDK 11 Release Notes:https://www.oracle.com/java/technologies/javase/11-relnote-issues.html
Java 11 – Features and Comparison:https://www.geeksforgeeks.org/java-11-features-and-comparison/