【51CTO.com快译】随着采用新的发布节奏,Java在2018年悄然经历了其开发过程中最大的变化之一。而这个大胆的新计划导致Java的开发人员每六个月发布一个新功能。

这有利于保持Java的新鲜度和相关性,但它很容易让开发人员错过引入的功能。本文汇总了几个有用的新功能并对其进行概述。

1.可选类(Optional class)

空指针异常是所有错误中最经典的错误之一。虽然开发人员可能很熟悉这个问题,但它的处理过程非常冗长,需要加以防范。至少在Java 8(以及Java 10改进)引入可选类之前是这样。

本质上,可选类允许包装一个变量,然后使用包装器的方法更简洁地处理空值。

清单1有一个多样性空指针错误的示例,其中的可选类引用foo为空(null),并且在其上访问了一个方法foo.getName()。

清单1.没有可选类的空指针

public class MyClass {     public static void main(String args[]) {       InnerClass foo = null;       System.out.println("foo = " + foo.getName());     } } class InnerClass {   String name = "";   public String getName(){       return this.name;   } }

可选类提供了多种处理此类情况的方法,具体取决于开发人员的需要。它运行一个isPresent()方法,开发人员可以使用它来进行if检查。然而这个过程可能相当冗长。但是可选类也有函数处理的方法。例如,清单2展示了如何使用ifPresent()——注意与isPresent()的一个字母差异,只有当存在数值时才运行输出代码。

清单2.只有当存在数值时运行代码

import java.util.Optional; public class MyClass {     public static void main(String args[]) {       InnerClass foo = null; //new InnerClass("Test");       Optional fooWrapper = Optional.ofNullable(foo);       fooWrapper.ifPresent(x -> System.out.println("foo = " + x.getName()));       //System.out.println("foo = " + fooWrapper.orElseThrow());     } } class InnerClass {   String name = "";   public InnerClass(String name){       this.name = name;   }   public String getName(){       return this.name;   } }

提示:当使用可选类时,如果使用orElse()方法通过方法调用提供默认值,需要考虑如果该值是非空的话,使用orElseGet()来提供函数引用,以获得不运行调用的性能优势。

2.记录类(预览功能)

构建Java应用程序的一个常见需求是所谓的不可变DTO(数据传输对象)。DTO用于对来自数据库、文件系统和其他数据存储的数据进行建模。传统上,DTO是通过创建一个类来创建的,该类的成员通过构造函数设置,没有getter方法来访问它们。Java 14引入并改进了Java 15,新的记录关键字为此目的提供了速记。

清单3说明了引入记录类型之前的典型DTO定义和用法。

清单3.一种简单的不可变DTO

public class MyClass {     public static void main(String args[]) {       Pet myPet = new Pet("Sheba", 10);        System.out.println(String.format("My pet %s is aged %s", myPet.getName(), myPet.getAge()));     } } class Pet {     String name;     Integer age;     public Pet(String name, Integer age){         this.name = name;         this.age = age;     }     public String getName(){         return this.name;     }     public Integer getAge(){         return this.age;     } }

可以使用记录关键字消除大部分样板文件,如清单4所示。

清单4.使用记录关键字

public class MyClass {     public static void main(String args[]) {       Pet myPet = new Pet("Sheba", 10);        System.out.println(String.format("My pet %s is aged %s", myPet.getName(), myPet.getAge()));     } }  public record Pet(String name, Integer age) {}

需要注意的是,使用数据对象的客户端代码没有改变;它的行为就像一个传统定义的对象。记录关键字足够智能,可以通过简单的定义足迹推断出存在哪些字段。

记录类还定义了equals()、hashCode()和toString()的默认实现,同时还允许开发人员覆盖这些实现。开发人员还可以提供自定义构造函数。

需要注意的是,记录不能被子类化。

3.新字符串方法

在Java 10和Java 12中,添加了几个有用的新字符串方法。除了字符串操作方法之外,还引入了两种用于简化文本文件访问的新方法。

Java 10中的新字符串方法:

  • isBlank():如果字符串为空或字符串仅包含空格(包括制表符),则返回true。注意isBlank()与isEmpty()不同,后者仅在length为0时返回true。

  • lines():将字符串拆分为字符串流,每个字符串包含一行。每行由/r或/n或/r/n定义。例如,参见清单5。

  • strip()、stripLeading()、stripTrailing():分别从开头和结尾、仅开头和仅结尾删除空格。

  • repeat(in ttimes):返回一个字符串,该字符串采用原始字符串并重复指定的次数。

  • readString():允许从文件路径直接读取字符串,如清单6所示。

  • writeString(Path path):将字符串直接写入指定路径的文件中。

Java 12中的新字符串方法:

  • indent(int level):将字符串缩进指定的数量。负值只会影响前导空格。

  • transform(Function f):将给定的lambda应用于字符串。

清单5. String.lines() 示例

import java.io.IOException; import java.util.stream.Stream; public class MyClass {     public static void main(String args[]) throws IOException{       String str = "test \ntest2 \n\rtest3 \r";       Stream lines = str.lines();       lines.forEach(System.out::println);     } }  /* outputs: test test2 test3 */

清单6.String.readString(Path path)示例

Path path = Path.of("myFile.txt");  String text = Files.readString(path); System.out.println(text);

4.Switch表达式

Java 12引入了Switch表达式,它允许在语句中内联使用Switch。换句话说,Switch表达式返回一个值。Java 12还提供了一种箭头语法,无需显式中断即可防止失败。Java 13则更进一步改进,引入了yield关键字来明确表示Switch case返回的值。Java 14采用了新的Switch表达式语法作为完整功能。

让我们看一些例子。首先,清单7有一个传统格式(Java 8)的Switch语句示例。此代码使用变量(消息)输出已知数字的名称。

清单7.传统Java Switch

class Main {    public static void main(String args[]) {     int size = 3;     String message = "";  switch (size){  case 1 : message = "one";  case 3 :    message = "three"; break;  default : message = "unknown"; break; }  System.out.println("The number is " + message);   } }

现在这段代码非常冗长并且挑剔。其实里面已经有了错误,开发人员需要仔细查找丢失的内容。清单8通过使用新Switch表达式进行了简化。

清单8. 新的Switch表达式

class NewSwitch {    public static void main(String args[]) {     int size = 3;      System.out.println("The number is " +       switch (size) {         case 1 -> "one";         case 3 -> "three";         default -> "unknown";       }     );   } }

在清单8中,可以看到Switch表达式就在System.out.println调用中。这已经是一个很大的可读性胜利,并且消除了多余的消息变量。此外,箭头语法通过消除break语句减少了代码占用空间。(不使用箭头语法时使用yield关键字。)

5.文本块

Java 13通过引入文本块解决了在Java中处理复杂文本字符串的长期困扰。Java 14改进了这种支持。

JSON、XML和SQL之类的东西可能会让开发人员过多地使用多个嵌套的转义层。正如规范解释的那样:“在Java中,在字符串文字中嵌入HTML、XML、SQL或JSON的片段……通常需要使用转义和连接进行大量编辑,然后才能编译包含该片段的代码。该代码段通常难以阅读且难以维护。”

在清单9中,新的文本块语法用于创建JSON片段。

清单9.使用文本块的JSON

class TextBlock {    public static void main(String args[]) {     String json = """       {         "animal" : "Quokka",         "link" : "https://en.wikipedia.org/wiki/Quokka"       }     """;      System.out.println(json);   } }

在清单9中看不到转义字符。此外,还要注意三重双引号语法。

6.密封类

Java 15(JEP 260)引入了密封类的概念。简而言之,新的sealed关键字允许开发人员定义哪些类可以对接口进行子类化。在这种情况下,示例胜过千言万语。参见清单10。

清单10.密封类示例

public abstract sealed class Pet     permits Cat, Dog, Quokka {...}

界面设计者在这里使用了sealed关键字来指定允许哪些类扩展Pet类。

总的来说,很明显,Java发布功能的新方法正在奏效。人们看到许多新想法通过JEP(JDK增强提案)过程转化为实际可用的Java功能。这对Java开发人员来说是一个好消息。这意味着他们正在使用一种充满活力、不断发展的语言和平台。

原文标题:6 great new Java features you don’t want to miss,作者:Matthew Tyson

【51CTO译稿,合作站点转载请注明原文译者和出处为51CTO.com】