> 文档中心 > 【函数式编程实战】(九) Optional实战大全

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

前言
📫 作者简介:小明java问道之路,专注于研究计算机底层,就职于金融公司后端高级工程师,擅长交易领域的高安全/可用/并发/性能的设计和架构📫 
🏆 Java领域优质创作者、阿里云专家博主、华为云享专家🏆
🔥 如果此文还不错的话,还请👍关注点赞收藏三连支持👍一下博主哦

目录

导读

一、用 Optional 取代null

1、为什么用 Optional 取代null

2、 使用Optional 优雅的解决null

二、Optional API大全


导读

本讲了解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 o = Optional.empty(); // 根据一个非空值创建 Optional对象 OrderInfo orderInfo = new OrderInfo(); Optional o1 = Optional.of(orderInfo); // 根据一个可以是空 Optional对象 orderInfo = null; Optional o2 = Optional.ofNullable(orderInfo);    }

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

    public static void main(String[] args) { // 使用map提取Optional对象中的值,map不可以连用 Optional o3 = Optional.ofNullable(orderInfo).map(OrderInfo::getOrderId); // flatMap方法:接受一个返回值为Optional的映射函数参数,该返回值亦是flatMap方法的返回值若结果为空,则返回 空Optional。 Optional<List> subOrderInfos = Optional.ofNullable(orderInfo)  .flatMap(orderInfo1 -> Optional.ofNullable(orderInfo1.getSubOrderInfoList())); // 获取子单的id,将Stream<Optional> 转化为 Stream List orderInfos = Arrays.asList(); List subOdrIdList = orderInfos.stream()  .map(OrderInfo::getSubOrderInfoList)  .flatMap(sub -> sub.stream())  .map(SubOrderInfo::getSubOrderIdOp)  .flatMap(Optional::stream)  .collect(Collectors.toList());    }    // 用 Optional封装对象    public Optional 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、ofNullable
of将指定值川optiona1封装之后返回,如果该值为null,则抛出一个NullPointerException
异常,ofNullable将指定值用 Optional封装之后返回,如果该值为null,则返回一个空的optional对象

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

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