Java10新特性解读

Java10是Java版本23年历史上最快的版本。Java因其缓慢的增长和进化而受到批评,但Java10打破了这一概念。Java10是一个具有许多未来变化的版本,其范围和影响可能并不明显,但却很牵强。

在本文中,我们将讨论Java10发行版中添加的各种特性。在此之前,让我们回顾一下java发布模型中引入的一些更改。

长期支持模式

从2017年开始,Oracle&Java社区宣布将向为期6个月的Java新节奏转变。它转向了oracle javase产品的长期支持(LTS)模型。

LTS版本的产品将提供甲骨文卓越和持续的支持,并将每3年瞄准一次。

每个Java发行版都是以一个或两个主要特性为模型的,这些特性驱动发行版。任何障碍都会推迟发布和推迟上市。Jigsaw项目是Java9的一个主要特性,它将发布日期推迟了几次,发布时间推迟了1.5年以上。6个月的cadence发布将遵循发布序列。放行列车每6个月有一个时间表。使切口上车的特征;否则它们等待下一列预定的火车。

Oracle JDK与Open JDK之比较

为了对开发人员更加友好,Oracle&Java社区现在将OpenJDK作为主要JDK来推广。这与早些时候相比是一个很大的解脱,当时JDK是适当的,并由Oracle授权,Oracle对重新分发有各种限制。不过,Oracle将继续生产JDK,但只针对长期支持版本。这是朝着更加云和容器友好的方向发展,因为开放的JDK库可以作为容器的一部分分发。

openjdk将每6个月发布一次,而Oracle JDK将每3年发布一次(LTS版本)。

将采用哪些JDK?

大型组织需要时间在不同的版本之间移动;他们会紧紧抓住版本直到他们能做到为止。行业对Java6的采用超过了Java7,然后行业逐渐转向Java8。在我看来,LTS版本将是最受企业青睐的。然而,究竟是oraclejdk的LTS版本还是openjdk还不清楚,部分原因是云计算领域正在进行大量工作。

Java9和Java10是非LTS版本。定于2018年9月发布的Java11将是LTS版本。

Java10功能

让我们来看看Java10中可用的特性。

基于时间的版本控制(JEP 322)

随着基于时间的发布周期的采用,Oracle改变了javase平台和JDK的版本字符串方案,以及相关的版本控制信息,用于当前和未来基于时间的发布模型。

版本号的新模式是:

$FEATURE.$INTERIM.$UPDATE.$PATCH

$FEATURE:counter将每6个月递增一次,并且将基于功能发布版本,例如:JDK 10、JDK 11。

$INTERIM: counter 对于包含兼容的错误修复和增强但没有不兼容更改的非功能版本,计数器将增加。通常,这将是零,因为六个月内不会有临时发布。这是为了将来对发布模型进行修订而保留的。

$UPDATE:counter对于修复新特性中的安全问题、退化和bug的兼容更新版本将增加。此功能发布后一个月更新,以后每3个月更新一次。2018年4月的版本是JDK10.0.1,7月的版本是JDK10.0.2,以此类推

$PATCH:counter将在紧急版本中增加,以修复关键问题。

添加了新的API以编程方式获取这些计数器值。让我们看看;

Version version = Runtime.version();
version.feature();
version.interim();
version.update();
version.patch();

现在,让我们看看返回版本信息的Java launcher:

$ java -version
java version "10" 2018-03-20
Java(TM) SE Runtime Environment 18.3 (build 10+46)
Java HotSpot(TM) 64-Bit Server VM 18.3 (build 10+46, mixed mode)

版本号格式是“10”,因为除了0之外没有其他计数器。添加发布日期。18.3可以理解为2018年和第3个月,版本10+46是版本10的第46个版本。对于JDK10.0.1的假设构建93,构建将是10.0.1+93

局部变量类型推断(JEP 286)

局部变量类型推断是Java10中为开发人员提供的最大的新特性。它将类型推断添加到带有初始值设定项的局部变量声明中。局部类型推断只能在以下情况下使用:

  • 仅限于具有初始值设定项的局部变量
  • 增强for循环的索引
  • 在for循环中声明的本地

我们来看看它的用法:

var numbers = List.of(1, 2, 3, 4, 5); // inferred value ArrayList<String>
// Index of Enhanced For Loop
for (var number : numbers) {
	System.out.println(number);
}
// Local variable declared in a loop
for (var i = 0; i < numbers.size(); i++) {
	System.out.println(numbers.get(i));
}

实验性基于Java的JIT编译器(JEP 317)

这个特性使基于Java的JIT编译器Graal能够在Linux/x64平台上作为一个实验性的JIT编译器使用。到目前为止,这是Java10特性列表中最具未来感的内容。

Graal是在java9中引入的。它是我们已经习惯的JIT编译器的替代品。它是JVM的一个插件,这意味着JIT编译器没有绑定到JVM,它可以动态地插入JVMCI兼容的任何其他插件(Java级JVM编译器接口)。它还带来了java世界中的提前编译(AOT)。它还支持多语言翻译。

“一个用Java编写的基于Java的实时编译器,用于将Java字节码转换为机器码。”这让人困惑吗?如果JVM是用Java编写的,那么您不需要JVM来运行JVM吗?JVM可以通过AOT编译,然后JIT编译器可以在jvmit中使用,通过实时代码优化来提高性能。

Graal是用Java从头开始的对JIT编译器的完全重写。以前的JIT编译器是用c++编写的。它被认为是任何编程语言进化的最后阶段。

您可以使用以下jvm参数切换到Graal:

-XX:+UnlockExperimentalVMOptions -XX:+UseJVMCICompiler

应用程序级数据共享(JEP 310)

此功能有助于改善启动占用空间,扩展现有的类数据共享(“CDS”)功能,以允许将应用程序类放置在共享存档中。

JVM在启动时执行一些初步步骤,其中一个步骤是在内存中加载类。如果有几个jar有多个类,那么第一个请求中的延迟就很明显了。这成为无服务器体系结构的一个问题,其中引导时间至关重要。为了缩短应用程序启动时间,可以使用应用程序类数据共享。其思想是通过在不同的Java进程之间共享公共类元数据来减少占用空间。可通过以下3个步骤实现:

确定要存档的类:使用java启动器创建要存档的文件列表,这可以通过以下参数实现:

$java -Xshare:off -XX:+UseAppCDS -XX:DumpLoadedClassList=hello.lst -cp hello.jar HelloWorld

创建AppCDS存档:使用java launcher创建要用于应用程序cd的文件列表的存档,这可以通过以下参数实现:

$java -Xshare:dump -XX:+UseAppCDS -XX:SharedClassListFile=hello.lst -XX:SharedArchiveFile=hello.jsa -cp hello.jar

使用AppCDS存档:使用带有以下参数的Java启动器来使用应用程序cd。

$java -Xshare:on -XX:+UseAppCDS -XX:SharedArchiveFile=hello.jsa -cp hello.jar HelloWorld

G1并行Full GC(JEP 307)

G1垃圾收集器在jdk9中是默认的。G1垃圾收集器避免了任何完全的垃圾收集,但是当用于收集的并发线程不能足够快地恢复内存时,用户的体验就会受到影响。

此更改通过使完全GC并行来改善G1最坏情况下的延迟。G1收集器的mark-sweep compact算法作为此更改的一部分被并行化,当用于收集的并发线程不能足够快地恢复内存时,它将被触发。

垃圾收集器接口(JEP 304)

这个JEP是未来的变化。它通过引入一个通用的垃圾收集器接口来改进不同垃圾收集器的代码隔离。

此更改为内部GC代码提供了更好的模块化。它将有助于将来添加新的GC而不改变现有的代码基,也有助于删除或管理以前的GC。

附加Unicode语言标记扩展(JEP 314)

此功能增强了java.util.Locale语言环境以及相关的API来实现BCP 47语言标记的额外Unicode扩展。从JavaSE9开始,支持的BCP47U语言标记扩展是“ca”和“nu”。此JEP将添加对以下附加扩展的支持:

  • cu(货币类型)
  • fw(每周第一天)
  • rg(区域覆盖)
  • tz(时区)

为了支持这些附加扩展,对各种api进行了更改,以提供基于U或附加扩展的信息。

java.text.DateFormat::get*Instance
java.text.DateFormatSymbols::getInstance
java.text.DecimalFormatSymbols::getInstance
java.text.NumberFormat::get*Instance
java.time.format.DateTimeFormatter::localizedBy
java.time.format.DateTimeFormatterBuilder::getLocalizedDateTimePattern
java.time.format.DecimalStyle::of
java.time.temporal.WeekFields::of
java.util.Calendar::{getFirstDayOfWeek,getMinimalDaysInWeek}
java.util.Currency::getInstance
java.util.Locale::getDisplayName
java.util.spi.LocaleNameProvider

根证书(JEP 319)

为了推广OpenJDK并使其对社区用户更具吸引力,此功能在JDK中提供了一组默认的根证书颁发机构(CA)证书。这也意味着Oracle和openjdk二进制文件在功能上是相同的。

关键的安全组件(如TLS)将在OpenJDK构建中默认工作。

Thread-Local 线程本地握手(JEP 312)

这是一个用于提高性能的内部JVM特性。

握手操作是在每个JavaThread处于safepoint状态时对其执行的回调。回调要么由线程本身执行,要么由VM线程执行,同时保持线程处于阻塞状态。

此功能提供了一种在线程上执行回调而不执行全局VM safepoint的方法。使停止单个线程,而不仅仅是停止所有线程或不停止任何线程既可能又便宜。

替代内存设备上的堆分配(JEP 316)

应用程序已经变得内存匮乏,云本地应用程序、内存数据库、流式应用程序都在增加。为了满足这些服务,有各种可用的内存体系结构。此功能增强了HotSpot VM在用户指定的备用内存设备(如NV-DIMM)上分配Java对象堆的能力。

此JEP针对具有与DRAM相同的语义(包括原子操作的语义)的替代内存设备,因此,可以在不更改现有应用程序代码的情况下代替DRAM用于对象堆。

删除Native-Header生成工具Javah(Jep313)

这是一个从JDK中删除javah工具的内务管理更改。javac中添加的工具功能是jdk8的一部分,它提供了在编译时编写Native-Header文件的能力,从而使javah变得无用。

将JDK林整合到单个存储库中(JEP 296)

多年来,在JDK代码库中有各种Mercurial存储库。不同的存储库确实提供了一些优势,但它们也有不同的操作缺点。作为此更改的一部分,JDK的许多存储库被合并到一个存储库中,以简化开发。

API更改

Java10已经添加和删除了API(是的,它不是一个拼写错误)。

Java9引入了增强的弃用,其中某些API被标记为在将来的版本中删除。

添加了API:Java10中添加了73个新API。

让我们看一些补充:

ListMapSet接口是通过静态copyOf(Collection)方法添加的。它返回一个不可修改的列表、映射或集合,其中包含所提供的条目。对于列表,如果给定的列表随后被修改,则返回的列表将不会反映这些修改。

Optional&它的原语变体获取一个方法orelsetrow()。这与get()完全相同,但是javadoc声明它是get()的首选替代方法

Collectors类获取用于收集不可修改集合(SetListMap)的各种方法

List<String> actors = new ArrayList<>();
actors.add("Jack Nicholson");
actors.add("Marlon Brando");
System.out.println(actors); // prints [Jack Nicholson, Marlon Brando]
// New API added - Creates an UnModifiable List from a List.
List<String> copyOfActors = List.copyOf(actors);
System.out.println(copyOfActors); // prints [Jack Nicholson, Marlon Brando]
// copyOfActors.add("Robert De Niro"); Will generate an
// UnsupportedOperationException
actors.add("Robert De Niro");
System.out.println(actors);// prints [Jack Nicholson, Marlon Brando, Robert De Niro]
System.out.println(copyOfActors); // prints [Jack Nicholson, Marlon Brando]

String str = "";
Optional<String> name = Optional.ofNullable(str);
// New API added - is preferred option then get() method
name.orElseThrow(); // same as name.get() 

// New API added - Collectors.toUnmodifiableList
List<String> collect = actors.stream().collect(Collectors.toUnmodifiableList());
// collect.add("Tom Hanks"); // Will generate an
// UnsupportedOperationException

以上就是Java10新特性总结的详细内容,更多关于Java10新特性的资料请关注我们其它相关文章!

(0)

相关推荐

  • Java13 明天发布(最新最全新特性解读)

    2017年8月,JCP执行委员会提出将Java的发布频率改为每六个月一次,新的发布周期严格遵循时间点,将在每年的3月份和9月份发布. 目前,JDK官网上已经可以看到JDK 13的进展,最新版的JDK 13将于2019年9月17日发布. 目前,JDK13处于Release-Candidate Phase(发布候选阶段),将于9月17日正式发布.目前该版本包含的特性已经全部固定,主要包含以下五个: JEP 350,Dynamic CDS Archives JEP 351,ZGC: Uncommit

  • Java8新特性:Lambda表达式之方法引用详解

    1.方法引用简述 方法引用是用来直接访问类或者实例的已经存在的方法或者构造方法.方法引用提供了一种引用而不执行方法的方式,它需要由兼容的函数式接口构成的目标类型上下文.计算时,方法引用会创建函数式接口的一个实例. 当Lambda表达式中只是执行一个方法调用时,不用Lambda表达式,直接通过方法引用的形式可读性更高一些.方法引用是一种更简洁易懂的Lambda表达式. Lambda表达式全文详情地址:http://blog.csdn.net/sun_promise/article/details/

  • Java8新特性时间日期库DateTime API及示例详解

    Java8新特性的功能已经更新了不少篇幅了,今天重点讲解时间日期库中DateTime相关处理.同样的,如果你现在依旧在项目中使用传统Date.Calendar和SimpleDateFormat等API来处理日期相关操作,这篇文章你一定不要错过.来刷新你的知识库吧! 背景 Java对日期.日历及时间的处理一直以来都饱受诟病,比如java.util.Date和java.util.Calendar类易用性差,不支持时区,非线程安全:还有用于格式化日期的类DateFormat也是非线程安全的等问题. J

  • 详细分析JAVA8新特性 Base64

    BASE64 编码是一种常用的字符编码,在很多地方都会用到.但base64不是安全领域下的加密解密算法.能起到安全作用的效果很差,而且很容易破解,他核心作用应该是传输数据的正确性,有些网关或系统只能使用ASCII字符.Base64就是用来将非ASCII字符的数据转换成ASCII字符的一种方法,而且base64特别适合在http,mime协议下快速传输数据. JDK里面实现Base64的API 在JDK1.6之前,JDK核心类一直没有Base64的实现类,有人建议用Sun/Oracle JDK里面

  • Java8新特性之新日期时间库的使用教程

    一.为什么引入新的日期时间库 Java对日期,日历及时间的处理一直以来都饱受诟病,尤其是它决定将java.util.Date定义为可修改的以及将SimpleDateFormat实现成非线程安全的. 关于这个新的时间日期库的最大的优点就在于它定义清楚了时间日期相关的一些概念,比方说,瞬时时间(Instant),持续时间(duration),日期(date),时间(time),时区(time-zone)以及时间段(Period).同时它也借鉴了Joda库的一些优点,比如将人和机器对时间日期的理解区分

  • java8新特性 stream流的方式遍历集合和数组操作

    前言: 在没有接触java8的时候,我们遍历一个集合都是用循环的方式,从第一条数据遍历到最后一条数据,现在思考一个问题,为什么要使用循环,因为要进行遍历,但是遍历不是唯一的方式,遍历是指每一个元素逐一进行处理(目的),而并不是从第一个到最后一个顺次处理的循环,前者是目的,后者是方式. 所以为了让遍历的方式更加优雅,出现了流(stream)! 1.流的目的在于强掉做什么 假设一个案例:将集合A根据条件1过滤为子集B,然后根据条件2过滤为子集C 在没有引入流之前我们的做法可能为: public cl

  • java8新特性之stream的collect实战教程

    1.list转换成list 不带return方式 List<Long> ids=wrongTmpList.stream().map(c->c.getId()).collect(Collectors.toList()); 带return方式 // spu集合转化成spubo集合//java8的新特性 List<SpuBo> spuBos=spuList.stream().map(spu -> { SpuBo spuBo = new SpuBo(); BeanUtils.c

  • Java8新特性Stream的完全使用指南

    什么是Stream Stream是Java 1.8版本开始提供的一个接口,主要提供对数据集合使用流的方式进行操作,流中的元素不可变且只会被消费一次,所有方法都设计成支持链式调用.使用Stream API可以极大生产力,写出高效率.干净.简洁的代码. 如何获得Stream实例 Stream提供了静态构建方法,可以基于不同的参数创建返回Stream实例 使用Collection的子类实例调用stream()或者parallelStream()方法也可以得到Stream实例,两个方法的区别在于后续执行

  • Java中JDK14的新特性之JFR,JMC和JFR事件流(推荐)

    简介 Java Flight Recorder(JFR)是JVM的诊断和性能分析工具.它可以收集有关JVM以及在其上运行的Java应用程序的数据.JFR是集成到JVM中的,所以JFR对JVM的性能影响非常小,我们可以放心的使用它. 一般来说,在使用默认配置的时候,性能影响要小于1%. JFR的历史很久远了.早在Oracle2008年收购BEA的时候就有了.JFR一般和JMC(Java Mission Control)协同工作. JFR是一个基于事件的低开销的分析引擎,具有高性能的后端,可以以二进

  • java8新特性 获取list某一列的操作

    提取某一列(以name为例) //从对象列表中提取一列(以name为例) List<String> nameList = studentList.stream().map(StudentInfo::getName).collect(Collectors.toList()); //提取后输出name nameList.forEach(s-> System.out.println(s)); 补充:java8新特性:lambda表达式:直接获得某个list/array/对象里面的字段集合 ja

  • 详解Java8新特性Stream之list转map及问题解决

    List集合转Map,用到的是Stream中Collectors的toMap方法:Collectors.toMap 具体用法实例如下: //声明一个List集合 List<Person> list = new ArrayList(); list.add(new Person("1001", "小A")); list.add(new Person("1002", "小B")); list.add(new Person

随机推荐