zoukankan      html  css  js  c++  java
  • Java 11 Tutorial

    Java 11 Tutorial

    参考 https://blog.csdn.net/sihai12345/article/details/82889827 

    原文 https://winterbe.com/posts/2018/09/24/java-11-tutorial/

    Java11已经发布了,我们今天聊聊大家还停留在哪个版本呢?大家对于新版本的迅速的发布有什么想说的呢?

    09 月 25 日,Oralce 正式发布了 Java 11,这是据 Java 8 以后支持的首个长期版本。

    我们都知道,前面的几个版本都不是长期支持的,然而,现在发布这个最新的长期支持的版本还是非常有意义的。

    Java11也有许多的地方增加了新的功能,当然,也有一些功能删除了。

    下面这是java版本的发布日期:

     


    java11也从下面的这些地方更新(下面是官网的截图)

    因为Java11已经把Java9和Java10的一些功能也集成到了Java11中,所以,下面我们从一些新的功能介绍一下。

    Java 9 -11 引入的新语法和API


    本地变量类型推断
    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 11";
    text = 23;  // Incompatible types

    您还可以final结合使用var禁止将变量重新分配给另一个值:

    final var text = "Banana";
    text = "Joe";   // Cannot assign a value to final variable 'text'

    当编译器无法推断出正确的变量类型时,也不允许使用var。 以下所有代码示例都会导致编译器错误:

    // Cannot infer type:
    var a;
    var nothing = null;
    var lambda = () -> 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;

    提示:在Intellij IDEA中,您可以将鼠标悬停在变量上,同时CMD/CTRL显示变量的感应类型(按键盘爱好者按下CTRL + J)。

    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

    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会通过发出警告。

    Streams
    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]

    如果您还不熟悉Streams,那么您应该阅读我的Java 8 Streams Tutorial

    Optionals
    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

    Strings
    Java11 给String增加了一些辅助方法来修剪或检查空格等功能:

    " ".isBlank();                // true
    " Foo Bar ".strip();          // "Foo Bar"
    " Foo Bar ".stripTrailing();  // " Foo Bar"
    " Foo Bar ".stripLeading();   // "Foo Bar "
    "Java".repeat(3);             // "JavaJavaJava"
    "A
    B
    C".lines().count();    // 3

    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);
    }

    这些上面的新特性只是在前面几个版本有的,或者一些比较觉得不错的新特性,如果还想去了解更多的新特性可以去官网查看(https://docs.oracle.com/en/java/javase/11/)

    或者查看下面的链接

    [用于反应式编程的Flow API]:https://community.oracle.com/docs/DOC-1006738
    [Java模块系统]:https://www.oracle.com/corporate/features/understanding-java-9-modules.html
    [应用程序类数据共享]:https://blog.codefx.org/java/application-class-data-sharing/
    [动态类文件常量]:http://openjdk.java.net/jeps/309
    [Java REPL(JShell)]:https://docs.oracle.com/javase/10/jshell/introduction-jshell.htm#JSHEL-GUID-630F27C8-1195-4989-9F6B-2C51D46F52C8
    [飞行记录器]:http://openjdk.java.net/jeps/328
    [Unicode 10]:http://openjdk.java.net/jeps/327
    [G1:完全并行垃圾收集器]:https://blog.idrsolutions.com/2018/04/java-10-improvements-to-garbage-collection-explained-in-5-minutes/
    [ZGC:可扩展的低延迟垃圾收集器]:http://openjdk.java.net/jeps/333
    [Epsilon:No-Op垃圾收集器]:http://openjdk.java.net/jeps/318
    [弃用Nashorn JavaScript引擎]:http://openjdk.java.net/jeps/335


    参考资料:
    https://docs.oracle.com/en/java/javase/11/
    http://openjdk.java.net/projects/jdk/11/
    https://winterbe.com/posts/2018/09/24/java-11-tutorial/
    https://www.oracle.com/technetwork/java/javase/11-relnote-issues-5012449.html#NewFeature

    =============== End

  • 相关阅读:
    干货分享!用户级爬虫,怎敢封IP
    一篇就够,网站数据提取,数据使用
    scrapy 解决爬虫IP代理池,数据轻松爬。
    使用代理ip的作用是什么?
    爬虫工作怎样选择代理ip
    爬取VIP视频
    python爬虫数据追加至excel中
    使用geany编辑器时python输出中文问题
    在Windows系统中从终端运行Python程序
    Windows 安装 Python 及配置环境变量
  • 原文地址:https://www.cnblogs.com/lsgxeva/p/10194281.html
Copyright © 2011-2022 走看看