在上一篇介绍Stream流式数据处理的文章中提到了Optional类,这是Java 8新增的一个类,用以解决程序中常见的NullPointerException异常问题。本篇文章将详细介绍Optional类,以及如何用它消除代码中的null检查。

避免使用null检查

作为Java开发人员,几乎所有人都遇到过NullPointerException异常,大多数人遇到NullPointerException异常时都会在异常出现的地方加上if代码块来判断值不为空,比如下面的代码:

public void bindUserToRole(User user) {if (user != null) {String roleId = user.getRoleId();if (roleId != null) {Role role = roleDao.findOne(roleId);if (role != null) {role.setUserId(user.getUserId());roleDao.save(role);}}}
}

这是比较普遍的做法,为了避免出现NullPointerException异常,手动对可能为null值进行了处理,不过代码看起来非常糟糕,业务逻辑被淹没在if逻辑判断中,也许下面的代码看起来可读性稍好一些:

public String bindUserToRole(User user) {if (user == null) {return;}String roleId = user.getRoleId();if (roleId == null) {return;}Role = roleDao.findOne(roleId);if (role != null) {role.setUserId(user.getUserId());roleDao.save(role);}
}

上面的代码避免了深层的if语句嵌套,但本质上是一样的,方法内有三个不同的返回点,出错后调试也不容易,因为你不知道是那个值导致了NullPointerException异常。

基于上面的原因,Java 8中引入了一个新的类Optional,用以避免使用null值引发的种种问题。

Optional类

java.util.Optional类是一个封装了Optional值的容器对象,Optional值可以为null,如果值存在,调用isPresent()方法返回true,调用get()方法可以获取值。

创建Optional对象

Optional类提供类三个方法用于实例化一个Optional对象,它们分别为empty()、of()、ofNullable(),这三个方法都是静态方法,可以直接调用。


empty()方法用于创建一个没有值的Optional对象:

Optional<String> emptyOpt = Optional.empty();

empty()方法创建的对象没有值,如果对emptyOpt变量调用isPresent()方法会返回false,调用get()方法抛出NullPointerException异常。


of()方法使用一个非空的值创建Optional对象:

String str = "Hello World";
Optional<String> notNullOpt = Optional.of(str);

ofNullable()方法接收一个可以为null的值:

Optional<String> nullableOpt = Optional.ofNullable(str);

如果str的值为null,得到的nullableOpt是一个没有值的Optional对象。

提取Optional对象中的值

如果我们要获取User对象中的roleId属性值,常见的方式是直接获取:

String roleId = null;
if (user != null) {roleId = user.getRoleId();
}

使用Optional中提供的map()方法可以以更简单的方式实现:

Optional<User> userOpt = Optional.ofNullable(user);
Optional<String> roleIdOpt = userOpt.map(User::getRoleId);

使用orElse()方法获取值

  • Optional类还包含其他方法用于获取值,这些方法分别为:

    • orElse():如果有值就返回,否则返回一个给定的值作为默认值;
    • orElseGet():与orElse()方法作用类似,区别在于生成默认值的方式不同。该方法接受一个Supplier<? extends T>函数式接口参数,用于生成默认值;
    • orElseThrow():与前面介绍的get()方法类似,当值为null时调用这两个方法都会抛出NullPointerException异常,区别在于该方法可以指定抛出的异常类型。
      下面来看看这三个方法的具体用法:
    String str = "Hello World";
    Optional<String> strOpt = Optional.of(str);
    String orElseResult = strOpt.orElse("Hello Shanghai");
    String orElseGet = strOpt.orElseGet(() -> "Hello Shanghai");
    String orElseThrow = strOpt.orElseThrow(() -> new IllegalArgumentException("Argument 'str' cannot be null or blank."));

    此外,Optional类还提供了一个ifPresent()方法,该方法接收一个Consumer<? super T>函数式接口,一般用于将信息打印到控制台:

    Optional<String> strOpt = Optional.of("Hello World");
    strOpt.ifPresent(System.out::println);

    使用filter()方法过滤

    filter()方法可用于判断Optional对象是否满足给定条件,一般用于条件过滤:

    Optional<String> optional = Optional.of("lw900925@163.com");
    optional = optional.filter(str -> str.contains("164"));

    在上面的代码中,如果filter()方法中的Lambda表达式成立,filter()方法会返回当前Optional对象值,否则,返回一个值为空的Optional对象。

如何正确使用Optional

通过上面的例子可以看出,Optional类可以优雅的避免NullPointerException带来的各种问题,不过,你是否真正掌握了Optional的用法?假设你试图使用Optional来避免可能出现的NullPointerException异常,编写了如下代码:

Optional<User> userOpt = Optional.ofNullable(user);
if (userOpt.isPresent()) {User user = userOpt.get();// do something...
} else {// do something...
}

坦白说,上面的代码与我们之前的使用if语句判断空值没有任何区别,没有起到Optional的正真作用:

if (user != null) {// do something...
} else {// do something...
}

当我们从之前版本切换到Java 8的时候,不应该还按照之前的思维方式处理null值,Java 8提倡函数式编程,新增的许多API都可以用函数式编程表示,Optional类也是其中之一。这里有几条关于Optional使用的建议:

  1. 尽量避免在程序中直接调用Optional对象的get()和isPresent()方法;
  2. 避免使用Optional类型声明实体类的属性;

第一条建议中直接调用get()方法是很危险的做法,如果Optional的值为空,那么毫无疑问会抛出NullPointerException异常,而为了调用get()方法而使用isPresent()方法作为空值检查,这种做法与传统的用if语句块做空值检查没有任何区别。

第二条建议避免使用Optional作为实体类的属性,它在设计的时候就没有考虑过用来作为类的属性,如果你查看Optional的源代码,你会发现它没有实现java.io.Serializable接口,这在某些情况下是很重要的(比如你的项目中使用了某些序列化框架),使用了Optional作为实体类的属性,意味着他们不能被序列化。

下面我们通过一些例子讲解Optional的正确用法:

正确创建Optional对象

上面提到创建Optional对象有三个方法,empty()方法比较简单,没什么特别要说明的。主要是of()和ofNullable()方法。当你很确定一个对象不可能为null的时候,应该使用of()方法,否则,尽可能使用ofNullable()方法,比如:

public static void method(Role role) {// 当Optional的值通过常量获得或者通过关键字new初始化,可以直接使用of()方法Optional<String> strOpt = Optional.of("Hello World");Optional<User> userOpt = Optional.of(new User());// 方法参数中role值不确定是否为null,使用ofNullable()方法创建Optional<Role> roleOpt = Optional.ofNullable(role);
}

orElse()方法的使用

return str != null ? str : "Hello World"
上面的代码表示判断字符串str是否为空,不为空就返回,否则,返回一个常量。使用Optional类可以表示为:

return strOpt.orElse("Hello World")

简化if-else

User user = ...
if (user != null) {String userName = user.getUserName();if (userName != null) {return userName.toUpperCase();} else {return null;}
} else {return null;
}

上面的代码可以简化成:

User user = ...
Optional<User> userOpt = Optional.ofNullable(user);return userOpt.map(User::getUserName).map(String::toUpperCase).orElse(null);

总结一下,新的Optional类让我们可以以函数式编程的方式处理null值,抛弃了Java 8之前需要嵌套大量if-else代码块,使代码可读性有了很大的提高。下一篇文章将介绍Java 8中新添加的日期API。

转载于:https://www.cnblogs.com/xzy-/p/10912769.html

Java 8新特性(三):Optional类相关推荐

  1. Java 8新特性:Optional类

    Java 8新特性:Optional类 在上一篇介绍Stream流式数据处理的文章中提到了Optional类,这是Java 8新增的一个类,用以解决程序中常见的NullPointerException ...

  2. 【Optional类】JDK1.8新特性之Optional类的使用

    这篇文章,主要介绍JDK1.8新特性之Optional类的使用. 目录 一.Optional类 1.1.Optional类介绍 1.2.Optional属性和方法 (1)value属性 (2)私有构造 ...

  3. Java 8 新特性,Optional介绍 | 春松客服

    目录 缘起 Optional类 Optional对象构造方法 提取Optional对象的值 使用orElse()方法获取值 使用filter()方法过滤 示例程序 注意事项 使用场景 orElseGe ...

  4. Java 16 新特性:record类

    以前我们定义类都是用class关键词,但从Java 16开始,我们将多一个关键词record,它也可以用来定义类.record关键词的引入,主要是为了提供一种更为简洁.紧凑的final类的定义方式. ...

  5. optional 解决空指针_Java8新特性:Optional类的正确使用姿势

    空指针异常是我们在实际开发中经常会遇到的问题,为了防止程序因为异常而中断,通常要在代码中添加大量的非空验证,例如一个释放 JDBC 相关资源的代码,如下所示. public static void r ...

  6. Java8新特性之Optional类(附代码案例)

    1. 概述 Optional< T >类是要给容器类,它可以保存类型T的值,代表这个值存在.或者仅仅保存null, 表示这个值不存在.原来用null表示一个值不存在,现在Optional可 ...

  7. JDK8新特性之Optional类

    Optional类 到目前为止,臭名昭著的空指针异常是导致Java应用程序失败的最常见原因.以前,为了解决空指针异常,Google公司著名的Guava项目引入了Optional类,Guava通过使用检 ...

  8. Java | 详解 Java 16 新特性 Record 记录类

    1 概述 Java Record 是一种不变类,类似于 String,Integer.让我们看一个简单的例子. public record Data( int x, int y) 我们使用 recor ...

  9. 【小家java】java8新特性之---Optional的使用,避免空指针,代替三目运算符

    相关阅读 [小家java]java5新特性(简述十大新特性) 重要一跃 [小家java]java6新特性(简述十大新特性) 鸡肋升级 [小家java]java7新特性(简述八大新特性) 不温不火 [小 ...

  10. 跟我学 Java 8 新特性之 Stream 流(三)缩减操作

    转载自   跟我学 Java 8 新特性之 Stream 流(三)缩减操作 和前面两篇文章一起服用,效果会更佳.通过对流API的基础体验Demo和关键知识点的讲解,相信大家对流API都有一定的认识了, ...

最新文章

  1. 探测参考信号(Sounding Reference Signal)
  2. Matplotlib的介绍及简单操作
  3. php超链接如何隐藏参数,php如何去除超链接
  4. php弹出第一个数组中,PHP array_search始终返回数组的第一个键
  5. 关于ANTLR的通用库的需求:使用反射来构建元模型
  6. iis运行原理 Asp.Net详解IIS内部运行原理
  7. async/await和Promise区别
  8. Javascript:一个屌丝的逆袭之路
  9. mysql免安装版配置百度_mysql -5.7.10 免安装版配置
  10. Android:自定义滚动边缘(EdgeEffect)效果
  11. 15 张 Vim 速查表奉上,帮你提高N倍效率!
  12. 【三维路径规划】基于matlab A_star算法无人机三维路径规划【含Matlab源码 003期】
  13. python 实现modBus协议的crc校验算法
  14. 组合数学(2)——组合矩阵
  15. qt调用vc编写库文件的方法
  16. phyton基础-01
  17. Mac 运行 JNLP 文件
  18. BGP路由器协议排错教程:BGP 对等体翻动问题
  19. java中的POJO是什么意思?
  20. Linux下USB刻录工具,Ubuntu上安装Etcher-开源USB刻录机工具

热门文章

  1. 【日常】计算机操作系统入门
  2. OpenJDK 和 Oracle JDK
  3. 【论文精读】AVP-Loc: Surround View Localization and Relocalization Based on HD VectorMap for AVP
  4. 乒乓操作的写入控制器
  5. Android撕衣服小案例
  6. 通过Unity2D独立开发一款瓷砖式RPG游戏需要学习哪些知识?
  7. 999句最常用英语口语
  8. JSON解析合集:你再也不用怕JSON解析了!!!
  9. 使用qrcode.js生成网址二维码
  10. 视差图转为深度图_纽劢研习社 | 深度图的非深度讲解