(转)Java 9~Java11的七大新特性
转载声明
本文原文为英文,作者Benjamin。本文内容转自高可用架构,由方圆翻译。
请大家关注公众号【高可用架构 】,尊重作者的辛苦付出。
本文转载只是为了方便学习。
如给原作者带来不便,请联系我删除。
0x01 摘要
Java11 发布了,然而很多公司还在用Java 8 ,本文会简要介绍Java 9 -11 引入的新语法和API。
0x02 本地变量类型推断
Java 10 就已经引入了新关键词var,该关键词可以在声明局部变量的时候替换类型信息。本地(local)是指方法内的变量声明。
Java 10之前,你需要这样声明一个String对象。
String text = "Hello Java 9"
在Java10里头可以使用var替代String,表达式变成这样:
var text = "Hello Java 10"
用var声明的变量仍然是静态类型的。 不兼容的类型无法重新分配给此类变量。 此代码段无法编译:
var text = "Hello Java 10"
text = 23; //类型不符合
当编译器无法推断出正确的变量类型时,也不允许使用var。 以下所有代码示例都会导致编译器错误:
// All CANNOT infer type:
var a;
var nothing = null;
var lambda = O -> System.out.println("Pity!");
var method = this: : someMethod;
局部变量类型推断可以泛型。 在下一个示例中,Map <String,List >类型,可以将其简化为单个var关键字,从而避免大量样板代码:
var myList = new ArrayList<Map<String, List<Integer>>>();
for (var current : myList) {
// current is infered to type: Map<String, List<Integer>>
System. out . println( current);
}
从Java 11开始,lambda参数也允许使用var关键字:
Predicate<String> predicate = (@Nullable var a) -> true;
0x03 HTTP Client
Java 9开始引入HttpClient API来处理HTTP请求。 从Java 11开始,这个API正式进入标准库包(java.net)。 让我们来探索一下我们可以用这个API做些什么。
新的HttpClient可以同步或异步使用。 同步请求会阻止当前线程。 BodyHandlers定义响应体的预期类型(例如,字符串,字节数组或文件):
var request = HttpRequest.newBuilder()
.uri(URI.create("https://winterbe.com"))
.GET()
.build();
var client = HttpClient.newHttpClient();
HttpResponse<String> response = client.send(request, HttpResponse.BodyHandlers.ofString());
System.out.println(response.body());
也可以使用异步来执行相同的请求。 调用sendAsync不会阻止当前线程,而是返回CompletableFuture来进行异步操作:
var request = HttpRequest.newBuilder()
.uri(URI.create("https://winterbe.com"))
.build();
var client = HttpClient.newHttpClient();
client.sendAsync(request, HttpResponse.BodyHandlers.ofString())
.thenApply(HttpResponse::body)
.thenAccept(System.out::println);
我们可以省略.GET,因为它是默认的请求方法。
下一个示例通过POST将数据发送到给定的URL。 与BodyHandler类似,您使用BodyPublishers定义作为请求主体发送的数据类型,如字符串,字节数组,文件或输入流:
var request = HttpRequest.newBuilder()
.uri(URI.create("https://postman-echo.com/post"))
.header("Content-Type", "text/plain")
.POST(HttpRequest.BodyPublishers.ofString("Hi there!"))
.build();
var client = HttpClient.newHttpClient();
var response = client.send(request, HttpResponse.BodyHandlers.ofString());
System.out.println(response.statusCode()); // 200
最后一个例子演示了如何通过BASIC-AUTH执行授权:
var request = HttpRequest.newBuilder()
.uri(URI.create("https://postman-echo.com/basic-auth"))
.build();
var client = HttpClient.newBuilder()
.authenticator(new Authenticator() {
@Override
protected PasswordAuthentication getPasswordAuthentication() {
return new PasswordAuthentication("postman", "password".toCharArray());
}
})
.build();
var response = client.send(request, HttpResponse.BodyHandlers.ofString());
System.out.println(response.statusCode()); // 200
0x04 Collections
List,Set和Map等集合已经用新方法扩展。 List.of从给定的参数创建了一个新的不可变列表。 List.copyOf创建列表的不可变副本:
var list = List.of("A", "B", "C");
var copy = List.copyOf(list);
System.out.println(list == copy); // true
因为list已经是不可变的,所以实际上不需要实际创建list实例的副本,因此list和副本是相同的实例。 但是,如果你复制一个可变list,那么复制确实会生成一个新实例,因此保证在改变原始list时没有副作用:
var list = new ArrayList<String>();
var copy = List.copyOf(list);
System.out.println(list == copy); // false
创建不可变map时,您不必自己创建map条目,而是将键和值作为参数传递:
var map = Map.of("A", 1, "B", 2);
System.out.println(map); // {B=2, A=1}
Java 11中的不可变集合仍然使用Collection API中的老接口。 但是,如果尝试修改不可变集合,则会抛出java.lang.UnsupportedOperationException。 可喜的是,如果尝试改变不可变集合,Intellij IDEA会发出警告。
0x05 Streams
Streams是在Java 8中引入的,Java 9增加了三个新方法。
- 单个参数构造方法:
Stream.ofNullable(null)
.count() // 0
增加 takeWhile 和 dropWhile 方法,接受predicate参数来决定用于从stream中丢弃的元素:
Stream.of(1, 2, 3, 2, 1)
.dropWhile(n -> n < 3)
.collect(Collectors.toList()); // [3, 2, 1]
Stream.of(1, 2, 3, 2, 1)
.takeWhile(n -> n < 3)
.collect(Collectors.toList()); // [1, 2]
如果对Stream不熟,可以参考Java 8 Streams Tutorial。
0x06 Optionals
Optionals也新增了一些用起来顺手的功能,例如 您现在可以简单地将Optional转换为Stream,或者为空的Optinal提供另一个Optional作为fallback
:
Optional.of("foo").orElseThrow(); // foo
Optional.of("foo").stream().count(); // 1
Optional.ofNullable(null)
.or(() -> Optional.of("fallback"))
.get(); // fallback
0x07 Strings
Java11 给String增加了一些辅助方法来trimming
或检查空格等功能:
" ".isBlank(); // true
" Foo Bar ".strip(); // "Foo Bar"
" Foo Bar ".stripTrailing(); // " Foo Bar"
" Foo Bar ".stripLeading(); // "Foo Bar "
"Java".repeat(3); // "JavaJavaJava"
"A\nB\nC".lines().count(); // 3
0x08 InputStreams
InputStream增加了transferTo方法,可以用来将数据直接传输到 OutputStream:
var classLoader = ClassLoader.getSystemClassLoader();
var inputStream = classLoader.getResourceAsStream("myFile.txt");
var tempFile = File.createTempFile("myFileCopy", "txt");
try (var outputStream = new FileOutputStream(tempFile)) {
inputStream.transferTo(outputStream);
}
0x09 其他的一些VM特性
从Java 8 到 Java 11引入了很多新特性,以下是这些特性的列表:
Flow API for reactive programming
Java Module System
Application Class Data Sharing
Dynamic Class-File Constants
Java REPL (JShell)
Flight Recorder
Unicode 10
G1: Full Parallel Garbage Collector
ZGC: Scalable Low-Latency Garbage Collector
Epsilon: No-Op Garbage Collector
Deprecate the Nashorn JavaScript Engine
译者注:对于译者来说还是Application Class-Data Sharing(CDS),ZGC和Flight Recorder比较有吸引力一点。关于ZGC,可以点击这里。