本博文将为您提供自Java 7以来增加的很棒的新功能的示例。我将展示每个Java版本的至少一项重大改进,一直到2020年秋季发布的Java 15都有。Java现在完全支持lambda和函数式编程,类型推断通过var,具有简单构造函数的不可变集合以及多行字符串。此外,还有令人兴奋的实验新功能,例如数据类(record)和sealed类。最后,我将讨论Java REPL,它为快速实验提供了很高的价值。
在Java 8中,功能编程和lambda被添加为语言功能。函数式编程的两个核心范例是不变值和将函数提升为一等公民的方法。数据经过一系列修改步骤,其中每个步骤都需要一些输入并将其映射到新的输出。函数式编程可与Java中的Streamsnull安全monads(Optional)一起使用,如下所示...
对于一般的计算机程序,通常必须使用值列表,并对每个值执行给定的转换。在Java 8之前,您必须使用for循环进行此转换,但是从现在开始,您可以使用Streams以下方法:
Stream.of("hello", "great") .map(s -> s + " world") .forEach(System.out::println); > hello world > great world
该map函数以一个lambda作为输入,它将应用于流中的所有元素。
Streams可以在Lists,Sets和Maps(通过转换)上工作。多亏了Streams,您可以摆脱代码中几乎所有的循环!
Java中的另一个常见问题是Null Pointer Exceptions。因此,Java引入了Optional –这是一个monad,它包装了一个可能为null或不为null的引用。可以通过函数性方式将更新应用于此Optional:
Optional.of(new Random().nextInt(10)) .filter(i -> i % 2 == 0) .map(i -> "number is even: " + i) .ifPresent(System.out::println); > number is even: 6
在上面的代码段中,我们创建一个随机数,将其包装在Optional对象中,然后仅打印偶数。
最后,我们有一个Java的REPL,它的名字叫JShell!简而言之,JShell允许在不编写和编译完整Java类的情况下尝试Java代码段。相反,您可以一次执行一个命令,然后立即看到结果。这是一个简单的例子:
$ <JDK>/bin/jshell jshell> System.out.println("hello world") hello world
长期以来,熟悉JavaScript或Python等解释型语言的人们都对REPL感到满意,但到目前为止,Java中缺少此功能。JShell允许定义变量,但也可以定义更复杂的实体,例如多行函数,类和执行循环。而且,JShell支持自动完成,如果您不知道给定Java类提供的确切方法,该功能将非常有用。
ListsJava的简单初始化早已丢失,但现在已经过去了。以前,您必须执行以下操作:
jshell> List<Integer> list = Arrays.asList(1, 2, 3, 4) list ==> [1, 2, 3, 4]
现在将其简化如下:
jshell> List<Integer> list = List.of(1, 2, 3, 4) b ==> [1, 2, 3, 4]
List,Set和Mapof(...)存在这种奇特的方法。它们都只用一行简单的代码就创建了一个不变的对象。
Java 10引入了新的var关键字,该关键字允许省略变量的类型。
jshell> var x = new HashSet<String>() x ==> [] jshell> x.add("apple") $1 ==> true
在上面的代码段中,编译器x可以将的类型推断为HashSet。
此功能有助于减少样板代码并提高可读性。但是,它有一些局限性:您只能var在方法主体内部使用,并且编译器将在编译时推断类型,因此所有内容仍为静态类型。
以前,当您编写了一个包含一个文件的简单Java程序时,必须先使用编译文件,javac然后使用来运行它java。在Java 11中,您可以使用一个命令完成两个步骤:
Main.java: public class Main { public static void main(String[] args) { System.out.println("hello world"); } } $ java ./Main.java hello world
对于仅由一个Java类组成的简单启动程序或实验,此用于启动单个源文件的功能将使您的生活更轻松。
Java 12为我们带来了Switch表达式。这是该表达式与旧的switch语句有何不同的快速展示。
在老switch语句定义了程序的流程:
jshell> var i = 3 jshell> String s; jshell> switch(i) { ...> case 1: s = "one"; break; ...> case 2: s = "two"; break; ...> case 3: s = "three"; break; ...> default: s = "unknown number"; ...> } jshell> s s ==> "three"
相反,新的switch表达式返回一个值:
jshell> var i = 3; jshell> var x = switch(i) { ...> case 1 -> "one"; ...> case 2 -> "two"; ...> case 3 -> "three"; ...> default -> "unknown number"; ...> }; x ==> "three"
总而言之,旧的switch语句用于程序流,新的switch表达式解析为一个值。
请注意,这个新的switch语句是一种映射函数:有一个输入(在上述情况下i),有一个输出(在此x)。这实际上是一种模式匹配功能,有助于使Java与哈四年编程原理更加兼容。一个类似的switch语句已经在斯卡拉推出已有一段时间。
需要注意的几件事:
代替双点,我们使用箭头 ->
不需要 break
考虑所有可能的情况时,可以省略默认情况
要在Java 12中启用此功能,请使用 --enable-preview --source 12
您是否曾经定义过长的多行字符串,例如JSON或XML?到目前为止,您可能已经将所有内容都压缩了一行并使用换行符\n,但这使String更加难以阅读。Java 13带有多行字符串!
案例:
public class Main { public static void main(String [] args) { var s = """ { "recipe": "watermelon smoothie", "duration": "10 mins", "items": ["watermelon", "lemon", "parsley"] }"""; System.out.println(s); } }
现在,我们通过单文件启动运行main方法:
java --enable-preview --source 13 Main.java { "recipe": "watermelon smoothie", "duration": "10 mins", "items": ["watermelon", "lemon", "parsley"] }
结果字符串跨越多行,引号""保持不变,甚至制表符\t也被保留!
在本文的所有新功能中,这可能是我最兴奋的功能:最后,Java中有数据类!这些类用record关键字声明,并具有自动Getter,构造函数和equals()方法等。总之,您可以摆脱大量的样板代码!
jshell> record Employee (String name, int age, String department) {} | created record Employee jshell> var x = new Employee("Anne", 25, "Legal"); x ==> Employee[name=Anne, age=25, department=Legal] jshell> x.name() $2 ==> "Anne"
Scala对于案例类具有类似的功能,对于Kotlin具有数据类具有类似的功能。到目前为止,在Java中,许多开发人员都使用Lombok,它提供了许多现在受recordsJava 14启发的功能。有关更多详细信息,请参见Baeldung文章。
Java的早期版本已经包含instanceof关键字:
Object obj = new String("hello"); if (obj instanceof String) { System.out.println("String length: " + ((String)obj).length()); }
不幸的部分:首先我们检查s类型是否为String,然后再次对其进行强制转换以获取其长度。
现在使用Java 14,编译器足够聪明,可以在instanceof check之后自动推断类型:
Object obj = new String("hello"); if (obj instanceof String mystr) { System.out.println("String length: " + mystr.length()); }
使用sealed关键字,您可以限制哪些类可以扩展给定的类或接口。这是一个例子:
public sealed interface Fruit permits Apple, Pear { String getName(); } public final class Apple implements Fruit { public String getName() { return "Apple"; } } public final class Pear implements Fruit { public String getName() { return "Pear"; } }
那么这对我们有什么帮助呢?好吧,现在您知道有多少个了Fruits。实际上,这是朝着完全支持的模式匹配的方向迈出的重要一步,在该模式下,您可以像对待枚举一样对待类。此sealed功能与switch前面说明的新表达式很好地结合在一起。
在过去的6年中,Java取得了长足的发展,此后实际上已经发布了8个Java新版本!与其他基于JVM的竞争对手(Scala和Kotlin)相比,所有这些令人敬畏的新功能有助于使Java成为竞争选择。
如果您正在寻找Java 8以后的Java新功能的更多详细信息,我可以推荐Andrew的DEV文章和David Csakvari的这篇文章。
以上就是java8到java15的新功能简介的详细内容,更多关于java 新功能的资料请关注小牛知识库其它相关文章!
主要内容:JEP 383 : 外部内存访问 API,JEP 339 : Edwards-Curve数字签名算法(EdDSA),JEP 373 : 重新实现旧版 DatagramSocket API,文本块作为标准,有用的空指针异常JEP 383 : 外部内存访问 API Java 14 允许 Java 程序安全有效地访问 Java 堆之外的外部内存。早期的 mapDB、memcached、ignite java 库提供了外部内存访问。它是一个更清晰的 API,可以无缝地操作所有类型的外部内存(本
主要内容:Java15弃用的功能,Java15移除的功能Java15弃用的功能 以下功能已弃用,并计划在未来版本中删除。 偏置锁定 : 默认情况下禁用偏置锁定,不推荐使用所有相关的命令行选项。JDK 社区正在寻找需要检查难以维护的偏向锁定的遗留同步机制。HotSpot 虚拟机中使用了偏置锁定来减少无竞争锁定的开销。 RMI 激活 :RMI 激活是自 Java 8 以来的可选部分,并且是 RMI 的过时部分。它正在强加持续的维护成本。 Java15移除的
本文向大家介绍Java8新增的重复注解功能示例,包括了Java8新增的重复注解功能示例的使用技巧和注意事项,需要的朋友参考一下 本文实例讲述了Java8新增的重复注解功能。分享给大家供大家参考,具体如下: 一 点睛 在Java 8以前,同一个程序元素前最多只能使用一个相同类型的Annotation;如果需要在同一个元素前使用多个相同类型的Annotation,则必须使用Annotation“容器”
San CLI UI核心功能分为六个部分,包括:项目的管理、配置管理、插件管理、依赖管理、内置构建任务以及仪表盘小工具。如图所示 整体功能可分为三类: 项目管理 工程构建 工具扩展
问题内容: 关于Java8 内置 功能接口,我遇到了许多问题,包括this,this和this。但是所有人都问“为什么只有一种方法?” 或“如果使用功能接口执行X,为什么会出现编译错误”等。我的问题是: 当我仍然可以在自己的接口中使用lambda时,这些新功能接口的存在目的 是 什么 ? 考虑一下oracle文档中的以下示例代码: 好的,很好,但这可以通过上面自己的示例来实现(带有单个方法的接口已
更新时间:2018-09-17 11:25:51 开发者在设备嵌入式C程序开发过程中需要准备本地的开发环境及工具,除了设备本身的开发,随着IoT开发场景和云端接入平台功能的不断丰富,相应的准备工作也在不断增加,例如下载接入云平台的特定SDK包,安装需要的依赖库,可能还需要为编译SDK使用特定版本的编译器及操作系统等。如果在准备本地环境过程中出现一些环境编译时的错误,还可能需要和合作方一起定位查找问