草庐IT

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

湘王 2023-04-18 原文

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

 

虽然到目前为止Java的版本更新还没有什么惊天动地的改变,但总是会冒出一些有趣的小玩意。前面列举了Java9Java10的一些特色,现在接着来撸一撸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中出现是个「早产儿」。

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

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

 

 

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

 

Java11有如下变更:

 

 

 

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

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

/**
 * 基于嵌套的访问控制
 * 
 * @author 湘王
 */
public class OuterClass {
    class InnerClass {
        public InnerClass() {
        }

        public void printOuterInt() {
            System.out.println("InnerClass");
        }
    }

    public static void main(String[] args) {
        Class<?> outerClass = OuterClass.class;
        // 得到宿主类
        Class<?> clazz1 = InnerClass.class.getNestHost();
        System.out.println(clazz1.getName());
        // 得到内部类成员
        Class<?>[] clazz2 = OuterClass.class.getNestMembers();
        for (Class<?> class1 : clazz2) {
            System.out.println(class1.getName());
            // 判断类是否是某个类的嵌套类
            System.out.println(outerClass.isNestmateOf(class1));
        }
    }
}

 

 

就是这样子用的。

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

/**
 * 飞行记录器
 *
 * @author 湘王
 */
public class FlightRecorder {
    @Label("Hello World")
    @Description("Helps the programmer getting started")
    static class HelloWorld extends Event {
        @Label("Message")
        String message;
    }

    public static void main(String[] args) {
        HelloWorld event = new HelloWorld();
        event.message = "hello, world!";
        event.commit();
    }
}

 

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

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

 

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

// 读取飞行记录数据
final Path path = Paths.get("C:\\recording.jfr");
List<RecordedEvent> recordedEvents;
try {
    recordedEvents = RecordingFile.readAllEvents(path);
    for (RecordedEvent event : recordedEvents) {
        System.out.println(event.getStartTime() + "," + event.getValue("message"));
    }
} catch (IOException e) {
    e.printStackTrace();
}

 

 

至于Java11中的其他更新项:

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

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

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

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

我个人觉得没啥特别的。

 

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

这是Java12的更新:

 

 

 

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

@SuppressWarnings("preview")
public static void main(String[] args) {
    DayOfWeek day = LocalDate.now().getDayOfWeek();

    int number = switch (day) {
        case MONDAY, WEDNESDAY, FRIDAY, SUNDAY -> 1;
        case TUESDAY, THURSDAY, SATURDAY -> 2;
    };
    System.out.println(number);

    switch (day) {
        case MONDAY:
            System.out.println("星期一");
            break;
        case TUESDAY:
            System.out.println("星期二");
            break;
        case WEDNESDAY:
            System.out.println("星期三");
            break;
        case THURSDAY:
            System.out.println("星期四");
            break;
        case FRIDAY:
            System.out.println("星期五");
            break;
        case SATURDAY:
            System.out.println("星期六");
            break;
        case SUNDAY:
            System.out.println("星期日");
            break;
    }
}

 

 

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

 

 

 

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

 

 

然后再敲代码:

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

 

 

Java14Java13基础上就更新的比较多了:

 

 

 

 

其中有一个可能会比较有用的特性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的话,那这项特性就没用了。

final class Point {
    public final int x;
    public final int y;
    public Point(int x, int y) {
        this.x = x;
        this.y = y;
    }
    // state-based implementations of equals, hashCode, toString
    // nothing else
}

 

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

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

 

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

 

record Name(String firstname, String lastname) { }

record Address(String... address) { }

record User(Name name, Address address, int age, ...) {
    static int x;
}

 

 

Java15Java14基础上又做了一系列更新:

 

 

 

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

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

 

 

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

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

 

 

 

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

 

 


 

 

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

 

有关Java新特性(2):Java 10以后的更多相关文章

  1. java - 等价于 Java 中的 Ruby Hash - 2

    我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/

  2. java - 从 JRuby 调用 Java 类的问题 - 2

    我正在尝试使用boilerpipe来自JRuby。我看过guide从JRuby调用Java,并成功地将它与另一个Java包一起使用,但无法弄清楚为什么同样的东西不能用于boilerpipe。我正在尝试基本上从JRuby中执行与此Java等效的操作:URLurl=newURL("http://www.example.com/some-location/index.html");Stringtext=ArticleExtractor.INSTANCE.getText(url);在JRuby中试过这个:require'java'url=java.net.URL.new("http://www

  3. java - 我的模型类或其他类中应该有逻辑吗 - 2

    我只想对我一直在思考的这个问题有其他意见,例如我有classuser_controller和classuserclassUserattr_accessor:name,:usernameendclassUserController//dosomethingaboutanythingaboutusersend问题是我的User类中是否应该有逻辑user=User.newuser.do_something(user1)oritshouldbeuser_controller=UserController.newuser_controller.do_something(user1,user2)我

  4. java - 什么相当于 ruby​​ 的 rack 或 python 的 Java wsgi? - 2

    什么是ruby​​的rack或python的Java的wsgi?还有一个路由库。 最佳答案 来自Python标准PEP333:Bycontrast,althoughJavahasjustasmanywebapplicationframeworksavailable,Java's"servlet"APImakesitpossibleforapplicationswrittenwithanyJavawebapplicationframeworktoruninanywebserverthatsupportstheservletAPI.ht

  5. Observability:从零开始创建 Java 微服务并监控它 (二) - 2

    这篇文章是继上一篇文章“Observability:从零开始创建Java微服务并监控它(一)”的续篇。在上一篇文章中,我们讲述了如何创建一个Javaweb应用,并使用Filebeat来收集应用所生成的日志。在今天的文章中,我来详述如何收集应用的指标,使用APM来监控应用并监督web服务的在线情况。源码可以在地址 https://github.com/liu-xiao-guo/java_observability 进行下载。摄入指标指标被视为可以随时更改的时间点值。当前请求的数量可以改变任何毫秒。你可能有1000个请求的峰值,然后一切都回到一个请求。这也意味着这些指标可能不准确,你还想提取最小/

  6. 【Java 面试合集】HashMap中为什么引入红黑树,而不是AVL树呢 - 2

    HashMap中为什么引入红黑树,而不是AVL树呢1.概述开始学习这个知识点之前我们需要知道,在JDK1.8以及之前,针对HashMap有什么不同。JDK1.7的时候,HashMap的底层实现是数组+链表JDK1.8的时候,HashMap的底层实现是数组+链表+红黑树我们要思考一个问题,为什么要从链表转为红黑树呢。首先先让我们了解下链表有什么不好???2.链表上述的截图其实就是链表的结构,我们来看下链表的增删改查的时间复杂度增:因为链表不是线性结构,所以每次添加的时候,只需要移动一个节点,所以可以理解为复杂度是N(1)删:算法时间复杂度跟增保持一致查:既然是非线性结构,所以查询某一个节点的时候

  7. 【Java入门】使用Java实现文件夹的遍历 - 2

    遍历文件夹我们通常是使用递归进行操作,这种方式比较简单,也比较容易理解。本文为大家介绍另一种不使用递归的方式,由于没有使用递归,只用到了循环和集合,所以效率更高一些!一、使用递归遍历文件夹整体思路1、使用File封装初始目录,2、打印这个目录3、获取这个目录下所有的子文件和子目录的数组。4、遍历这个数组,取出每个File对象4-1、如果File是否是一个文件,打印4-2、否则就是一个目录,递归调用代码实现publicclassSearchFile{publicstaticvoidmain(String[]args){//初始目录Filedir=newFile("d:/Dev");Datebeg

  8. java - 为什么 ruby​​ modulo 与 java/other lang 不同? - 2

    我基本上来自Java背景并且努力理解Ruby中的模运算。(5%3)(-5%3)(5%-3)(-5%-3)Java中的上述操作产生,2个-22个-2但在Ruby中,相同的表达式会产生21个-1-2.Ruby在逻辑上有多擅长这个?模块操作在Ruby中是如何实现的?如果将同一个操作定义为一个web服务,两个服务如何匹配逻辑。 最佳答案 在Java中,模运算的结果与被除数的符号相同。在Ruby中,它与除数的符号相同。remainder()在Ruby中与被除数的符号相同。您可能还想引用modulooperation.

  9. java - Ruby 相当于 Java 的 Collections.unmodifiableList 和 Collections.unmodifiableMap - 2

    Java的Collections.unmodifiableList和Collections.unmodifiableMap在Ruby标准API中是否有等价物? 最佳答案 使用freeze应用程序接口(interface):Preventsfurthermodificationstoobj.ARuntimeErrorwillberaisedifmodificationisattempted.Thereisnowaytounfreezeafrozenobject.SeealsoObject#frozen?.Thismethodretur

  10. 由于 libgmp.10.dylib 的问题,Ruby 2.2.0 无法运行 - 2

    我刚刚安装了带有RVM的Ruby2.2.0,并尝试使用它得到了这个:$rvmuse2.2.0--defaultUsing/Users/brandon/.rvm/gems/ruby-2.2.0dyld:Librarynotloaded:/usr/local/lib/libgmp.10.dylibReferencedfrom:/Users/brandon/.rvm/rubies/ruby-2.2.0/bin/rubyReason:Incompatiblelibraryversion:rubyrequiresversion13.0.0orlater,butlibgmp.10.dylibpro

随机推荐