• Java新特性(2):Java 10以后


    您好,我是湘王,这是我的CSDN博客,欢迎您来,欢迎您再来~

    虽然到目前为止Java的版本更新还没有什么惊天动地的改变,但总是会冒出一些有趣的小玩意。前面列举了Java9和Java10的一些特色,现在接着来撸一撸Java11之后的新奇特。

    Java9更新了Http 2 Client,也说过先不着急看,因为在后续版本中语法会变。这不,到Java11就实现了。最直接的变化就是http相关包名由Java 9的jdk.incubator.http改为Java 11的java.net.http。感觉java.net.http才像那么回事,incubator是个啥呢?好像完全和http不沾边。本着好奇害死猫的精神,查了下incubator的意思:

     

    好吧,原来JDK工作组认为http在Java9中出现是个「早产儿」。

    再来看看Java11对http的更新,例如通过http访问某度的主页:

    1. // 包名由Java 9的jdk.incubator.http改为Java 11的java.net.http
    2. HttpClient client = HttpClient.newHttpClient();
    3. HttpRequest request = HttpRequest.newBuilder()
    4. .uri(URI.create("http://www.baidu.com/"))
    5. .build();
    6. client.sendAsync(request, BodyHandlers.ofString())
    7. .thenApply(HttpResponse::body)
    8. .thenAccept(System.out::println)
    9. .join();

    (现在Lambda表达式几乎无处不在了,如果还不会的话搞不好以后Java代码都看不懂了)

    Java11有如下变更:

     

    除了http之外,比较有用的就两个:基于嵌套的访问控制(181)和飞行记录器(328)。

    所谓基于嵌套的访问控制,其实就是能够判断某个类是不是另一个类的嵌套类。

    1. /**
    2. * 基于嵌套的访问控制
    3. *
    4. * @author 湘王
    5. */
    6. public class OuterClass {
    7. class InnerClass {
    8. public InnerClass() {
    9. }
    10. public void printOuterInt() {
    11. System.out.println("InnerClass");
    12. }
    13. }
    14. public static void main(String[] args) {
    15. Class outerClass = OuterClass.class;
    16. // 得到宿主类
    17. Class clazz1 = InnerClass.class.getNestHost();
    18. System.out.println(clazz1.getName());
    19. // 得到内部类成员
    20. Class[] clazz2 = OuterClass.class.getNestMembers();
    21. for (Class class1 : clazz2) {
    22. System.out.println(class1.getName());
    23. // 判断类是否是某个类的嵌套类
    24. System.out.println(outerClass.isNestmateOf(class1));
    25. }
    26. }
    27. }

    就是这样子用的。

    所谓飞行记录器,就是模仿飞机上的黑匣子,是一种低开销的事件信息收集框架,它原来是JDK商业版(是一种大厂之间合作收费的版本)中的一项分析工具,主要数据源于应用程序、JVM和OS,是在故障发生后,能够从事件记录文件中提取出有用信息对故障进行分析。到了Java11,它索性就免费了。

    1. /**
    2. * 飞行记录器
    3. *
    4. * @author 湘王
    5. */
    6. public class FlightRecorder {
    7. @Label("Hello World")
    8. @Description("Helps the programmer getting started")
    9. static class HelloWorld extends Event {
    10. @Label("Message")
    11. String message;
    12. }
    13. public static void main(String[] args) {
    14. HelloWorld event = new HelloWorld();
    15. event.message = "hello, world!";
    16. event.commit();
    17. }
    18. }

    运行上述代码时加上JVM参数:

    -XX:StartFlightRecording=duration=1s, filename=C:\\recording.jfr

    然后再通过飞行记录器来读取数据:

    1. // 读取飞行记录数据
    2. final Path path = Paths.get("C:\\recording.jfr");
    3. List recordedEvents;
    4. try {
    5. recordedEvents = RecordingFile.readAllEvents(path);
    6. for (RecordedEvent event : recordedEvents) {
    7. System.out.println(event.getStartTime() + "," + event.getValue("message"));
    8. }
    9. } catch (IOException e) {
    10. e.printStackTrace();
    11. }

    至于Java11中的其他更新项:

    1、Epsilon GC(no-op GC,318)

    2、ZGC可伸缩低延迟垃圾收集器(333)

    3、支持TLSv1.3协议(332)

    4、动态类文件常量(309)

    我个人觉得没啥特别的。

    Java11之后,感觉Java是为了更新而更新,新奇特不多。差不多每个版本多那么一个小玩意。

    这是Java12的更新:

     

    Java12多了一个switch表达式(325):

    1. @SuppressWarnings("preview")
    2. public static void main(String[] args) {
    3. DayOfWeek day = LocalDate.now().getDayOfWeek();
    4. int number = switch (day) {
    5. case MONDAY, WEDNESDAY, FRIDAY, SUNDAY -> 1;
    6. case TUESDAY, THURSDAY, SATURDAY -> 2;
    7. };
    8. System.out.println(number);
    9. switch (day) {
    10. case MONDAY:
    11. System.out.println("星期一");
    12. break;
    13. case TUESDAY:
    14. System.out.println("星期二");
    15. break;
    16. case WEDNESDAY:
    17. System.out.println("星期三");
    18. break;
    19. case THURSDAY:
    20. System.out.println("星期四");
    21. break;
    22. case FRIDAY:
    23. System.out.println("星期五");
    24. break;
    25. case SATURDAY:
    26. System.out.println("星期六");
    27. break;
    28. case SUNDAY:
    29. System.out.println("星期日");
    30. break;
    31. }
    32. }

    Java13整体上与Java12差不多,没什么变化:

     

    唯一比较引人注目的就是模仿Python搞了一个三引号的文本块(而且之前刚出来的时候还只能在idea中实验):

     

    然后再敲代码:

    1. @SuppressWarnings("preview")
    2. public static void main(String[] args) {
    3. // 文本块
    4. String text = """
    5. Lorem ipsum dolor sit amet, consectetur adipiscing
    6. elit, sed do eiusmod tempor incididunt ut labore
    7. et dolore magna aliqua.
    8. """;
    9. System.out.println(text);
    10. }

    Java14在Java13基础上就更新的比较多了:

     

    其中有一个可能会比较有用的特性:NullPointerExceptions增强(358):

    例如码农可能会写这样的代码:

    通常会有这样的代码:

    User user = new User();

    String cityname = user.getDetailInfo().getAddress().getCity().getName();

    System.out.println(cityname);

    在调用过程中,如果User、DetailInfo、Address、City中有任何一个是null,那么就会抛出NullPointerExceptions,但是比较难于确定倒底是哪一个对象为null。打印出来的异常信息是:

     

    但在Java14中,却可以很准确地知道NPE发生在哪里:

    先通过JVM参数打开这项特性:-XX:+ShowCodeDetailsInExceptionMessages

     

    再次运行相同的代码,可以看到打印出的异常信息变了:

     

    不过有个条件:如果代码中已有捕获的NullPointerExceptions,那么就不会执行异常计算。也就是说如果使用了try...catch的话,那这项特性就没用了。

    Java14另一个比较有用的特性是Record类型。对,你没看错,不是类,是类型,是和Integer、String一样的类型,叫Record。那什么是Record类型呢?

    以前在定义一个JavaBean的时候,通常都会附带定义一些构造函数、getter/setter、equals()、hashCode()以及toString()等无用且无聊的代码。所以出现了第三方类库Lombok,它就可以自动生成这些代码。例如:

    1. final class Point {
    2. public final int x;
    3. public final int y;
    4. public Point(int x, int y) {
    5. this.x = x;
    6. this.y = y;
    7. }
    8. // state-based implementations of equals, hashCode, toString
    9. // nothing else
    10. }

    Record类型就是为了定义这种「纯数据载体」而生的:

    1. record Name(String firstname, String lastname) { }
    2. record Address(String... address) { }
    3. record User(Name name, Address address, int age, ...) {
    4. static int x;
    5. }

    Java15在Java14基础上又做了一系列更新:

     

    Java15比较突出的是增加一个称之为「封印类」(360,特性编号也是360,不会是巧合吧)的安全类。所谓封印类,其实就是告诉外界,只有哪些类或接口可以继承或实现它。

    1. // 这段代码声明了一个Shape接口,而permits列表限制了只有「Circle」和「Rectangle」可以实现Shape
    2. // 任何其他尝试扩展Shape的类或接口都将收到编译错误
    3. sealed interface Shape permits Circle, Rectangle {
    4. }

    封印类常常和record一起使用:

    1. // 只有「Circle」和「Rectangle」可以实现Shape
    2. // 「Circle」需要一个坐标和半径才能确定
    3. // 「Rectangle」需要两个坐标就能确定
    4. sealed interface Shape permits Circle, Rectangle {
    5. record Circle(Point center, int radius) implements Shape { }
    6. record Rectangle(Point lowerLeft, Point upperRight) implements Shape { }
    7. }

    后续的Java更新就不再啰嗦了。还是之前的那个态度:Java8是一个非常重要的分水岭,如果不把Lambda表达式和函数式编程学好,后面可能会有很多骚操作都做不了了。


    感谢您的大驾光临!咨询技术、产品、运营和管理相关问题,请关注后留言。欢迎骚扰,不胜荣幸~

  • 相关阅读:
    FFmpeg入门详解之103:FFmpeg Nginx VLC打造M3U8直播点播
    以sqlilabs靶场为例,讲解SQL注入攻击原理【42-50关】
    设计模式(2) - 创建型模式
    汇编 80x86 计算机组织
    内网安全学习
    CentOS7 安装 Nacos过程
    2022-04-10-Docker
    MyBatis多条件查询、动态SQL、多表操作、注解开发详细教程
    基于聚类和回归分析方法探究蓝莓产量影响因素与预测模型研究附录
    C++ day3
  • 原文地址:https://blog.csdn.net/lostrex/article/details/127912039