【精通函数式编程】(九) Optional实战大全

举报
小明的混沌之路 发表于 2022/07/31 13:50:17 2022/07/31
【摘要】 本讲了解null的缺陷,了解为什么用Optional 取代null,学习Optional 的必要性,代码里怎么用Optional 才能更优雅


前言📫 作者简介:小明java问道之路,专注于研究计算机底层,就职于金融公司后端高级工程师,擅长交易领域的高安全/可用/并发/性能的设计和架构📫 

🏆 Java领域优质创作者、阿里云专家博主、华为云专家🏆

🔥 如果此文还不错的话,还请👍关注点赞收藏三连支持👍一下博主哦

导读

本讲了解null的缺陷,了解为什么用 Optional 取代null,学习Optional 的必要性,代码里怎么用Optional 才能更优雅

一、用 Optional 取代null

用 Optional 取代null总要有原因的,我们先看看null的缺陷,了解为什么用 Optional 取代null,代码里怎么用Optional 才能更优雅

1、为什么用 Optional 取代null

null的缺陷本身会造成很严重的问题,我们熟知的NullPointerException是目前Java以及其他语言最典型的异常,当然我们可以避免,就是在每个对象使用的时候判空,例如 if(null != object) ,虽然可以避免NullPointerException,但是代码想膏药一样去不掉,可读性很差,代码里充斥大量判空处理

null本身是没有意义的,我们一般接口返回会有 respMsg、respCode、result等等,返回null代表出现了错误,是需要进行异常处理的

null破坏了Java的设计原理,在Java中屏蔽了底层的指针操作,但是唯一例外的就是空指针异常

2、 使用Optional 优雅的解决null

Java吸取了Scala等语言,在Java8中引入了 java.util.Optional,该类封装了Optional类型的值,可以接收泛型类型的值,可以理解为,将 所需要的对象包在 Optional对象中,变量存在时,Optional只是封装所需对象,不存在时,缺失的值会被构建成一个空的Optional对象,这个对象有Optional.empty() 方法返回

Optional.empty() 方法是一个静态工程,它会返回一个Optional类的特定单一实例,如果你尝试解析一个空对象一定会报一个NullPointerException,但是使用Optional.empty() 方法,是解析一个有效的Optional对象,不对是你所需的对象是空的

在你的代码中建议使用Optional对象接收对象,这样不需要进行判空等操作,可以清晰的判断数据问题还是某些设计问题,引入Optional对象也不是消除所有的null,而是更好的管理对象

​创建Optional对象有几种形式:

public static void main(String[] args) {
    // 创建一个空的 Optional对象,但是这个对象并不会报java.lang.NullPointerException
    Optional<OrderInfo> o = Optional.empty();

    // 根据一个非空值创建 Optional对象
    OrderInfo orderInfo = new OrderInfo();
    Optional<OrderInfo> o1 = Optional.of(orderInfo);

    // 根据一个可以是空 Optional对象
    orderInfo = null;
    Optional<OrderInfo> o2 = Optional.ofNullable(orderInfo);
}

创建出来的Optional对象,有什么Java提供的API,我们如何从Optional对象中提取我们需要的值,如何使用Optional对象构成Stream流呢

public static void main(String[] args) {
    // 使用map提取Optional对象中的值,map不可以连用
    Optional<String> o3 = Optional.ofNullable(orderInfo).map(OrderInfo::getOrderId);

    // flatMap方法:接受一个返回值为Optional的映射函数参数,该返回值亦是flatMap方法的返回值若结果为空,则返回 空Optional。
    Optional<List<SubOrderInfo>> subOrderInfos = Optional.ofNullable(orderInfo)
            .flatMap(orderInfo1 -> Optional.ofNullable(orderInfo1.getSubOrderInfoList()));

    // 获取子单的id,将Stream<Optional<String>> 转化为 Stream<String>
    List<OrderInfo> orderInfos = Arrays.asList();
    List<String> subOdrIdList = orderInfos.stream()
            .map(OrderInfo::getSubOrderInfoList)
            .flatMap(sub -> sub.stream())
            .map(SubOrderInfo::getSubOrderIdOp)
            .flatMap(Optional::stream)
            .collect(Collectors.toList());
}

// 用 Optional封装对象
public Optional<String> getSubOrderIdOp() {
    return Optional.ofNullable(subOrderIdOp);
}

二、Optional API大全

empty 返回一个空的 Optional 实例


filter如果值存在并且满足提供的谓词(条件),就返回包含该值的 Optional 对象,否则返回一个空的Optional对象


flatMap、map如果值存在,就对该值执行提供的 mapping 函数调用,返回一个 Optional 类型的值,否则就返回一个空的Optional对象。map,如果值存在,就对该值执行提供的 mapping 函数调用


get如果值存在,就将该值用 optional 封装返回,否则抛出一个 NoSuchElement Exception异常


ifPresentOrElse、isPresent如果值存在,就以值作为输人执行对应的方法调用,否则执行另一个不需任何输人的方法。isPresent,如果值存在就返回 true,否则返回 talse


of、ofNullableof将指定值川optiona1封装之后返回,如果该值为null,则抛出一个NullPointerException异常,ofNullable将指定值用 Optional封装之后返回,如果该值为null,则返回一个空的optional对象


or、orElse 、orElseGet 、orElseThrow or 如果值存在,就返回同一个optional对象,否则返回由支持函数生成的另一个optional对象。orElse 如果有值则将其返回,否则返回一个默认值。orElseGet 如果有值则将其返回,否则返回一个由指定的 Supplier 接口生成的值。orElseThrow 如果有值则将其返回,否则抛出一个由指定的 Supplier接口生成的异常


stream 如果有值,就返回包含该值的一个Stream,否则返回一个空的 Stream

【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。