Java11已经发布了,我们今天聊聊大家还停留在哪个版本呢?大家对于新版本的迅速的发布有什么想说的呢?
09 月 25 日,Oralce 正式发布了 Java 11,这是据 Java 8 以后支持的首个长期版本。
我们都知道,前面的几个版本都不是长期支持的,然而,现在发布这个最新的长期支持的版本还是非常有意义的。
Java11也有许多的地方增加了新的功能,当然,也有一些功能删除了。
下面这是java版本的发布日期:
5954965-80173492f7a44cc1.jpg
java11也从下面的这些地方更新(下面是官网的截图)
搜狗截图20180928195041.jpg" src="https://upload-images.jianshu.io/upload_images/5824016-85d2b969797bf0ae.jpg?imageMogr2/auto-orient/strip%7C/Upload/Images/2018100301/3A680A484E2C3EB2.jpg" style="font-size: inherit; color: inherit; line-height: inherit; padding: 0px; display: block; margin: 0px auto; max-width: 100%;" alt="搜狗截图20180928195041.jpg">搜狗截图20180928195041.jpg 搜狗截图20180928195105.jpg
因为Java11已经把Java9和Java10的一些功能也集成到了Java11中,所以,下面我们从一些新的功能介绍一下。
Java 10 就已经引入了新关键词var,该关键词可以在声明局部变量的时候替换类型信息。本地(local)是指方法内的变量声明。
Java 10之前,你需要这样声明一个String对象。
class="hljs javascript" style="margin: 0px 2px; line-height: 18px; font-size: 14px; font-weight: normal; letter-spacing: 0px; font-family: Consolas, Inconsolata, Courier, monospace; border-radius: 0px; padding: 0.5em; color: #f8f8f2; display: block !important; white-space: pre !important; overflow: auto !important;">String?str="hello?java?9";
在Java10里头可以使用var替代String,表达式变成这样:
var?str="hello?java?10";
用var声明的变量仍然是静态类型的。 不兼容的类型无法重新分配给此类变量。 此代码段无法编译:
var?str="hello?java?11";
str=11;??//Incompatible?types
当编译器无法推断出正确的变量类型时,也不允许使用var。 以下所有代码示例都会导致编译器错误:
//?Cannot??????infer?type:
var?a;
var?nothing?=null;
var??lambda=()->System.out.prinltn("Pity!");
var?method=this::someNethod;
局部变量类型推断可以泛型。 在下一个示例中,Map 类型,可以将其简化为单个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?=?function" style="font-size: inherit; color: inherit; line-height: inherit; margin: 0px; padding: 0px;">(@Nullable?var?a)->true;
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
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
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会通过发出警告。
Streams是在Java 8中引入的,Java 9增加了三个新方法。 单个参数构造方法:
Stream.ofNullable(null)
????.count()???//?0
增加 takeWhile 和 dropWhile 方法,用于从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不熟,可以参考这篇文章[1]。
Optionals提供了一些非常方便的功能,例如 您现在可以简单地将Optional转换为Stream,或者为空Optinal提供另一个Optional作为备胎:
Optional.of("foo").orElseThrow();?????//?foo
Optional.of("foo").stream().count();??//?1
Optional.ofNullable(null)
????.or(()?->?Optional.of("fallback"))
????.get();???????????????????????????//?fallback
Java11 给String增加了一些辅助方法来修剪或检查空格等功能:
"?".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
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);
}
这些上面的新特性只是在前面几个版本有的,或者一些比较觉得不错的新特性,如果还想去了解更多的新特性可以去官网查看(https://docs.oracle.com/en/java/javase/11/)
或者查看下面的链接
更多请扫描下方的二维码关注我们的微信公众号,每日推送新鲜资讯干货!
关注微信公众号“好好学java”,干货每日更新!
?