摘要:介绍Stream的容器类中 Optional 方法。

综述

  仰慕 Stream 流久矣,终于有机会彻底的了解其特性以及用法了,关于源码的理解还需要持续增加深度。在学习 Stream 的时候,同时认识了强大的 Optional,本文就介绍 Optional 的API。

  Optional 是Java 8提供的一个容器类,用来装载可能为空的引用。作为java程序员,经常在业务逻辑中检查对象是否为空,引入Optional可以很友好地解决空指针异常,避免显式进行空值检测,提高代码的可读性和可维护性。

Optional类的API

Optional of(T t)

  使用静态方法 of(T t) 创建一个非空的 Optional 对象。注意:传递的参数t必须是非空的,也就是说不能为 null;否则,抛出空指针异常NullPointerException。

Optional ofNullable(T t)

  使用静态方法 ofNullable(T t) 创建一个即可空又可非空的 Optional 对象。

  Optional可以通过工厂方法of()、ofNullable()和empty()创建实例,示例如下:

// 创建非空的Optional实例
Optional<String> optional = Optional.of(“hello”);
// 创建值为空的的Optional对象
Optional<String> optOrNull = Optional.ofNullable(null);
System.out.println(optOrNull); // 输出:Optional.empty
// 使用静态方法empty()创建空的Optional实例
Optional<String> empty = Optional.empty();

boolean isPresent()

  isPresent() 用于判断一个 Optional 对象是否存在值,如果存在,该方法返回 true;否则,返回 false。Java 11 后还可以通过方法 isEmpty() 判断与 isPresent() 相反的结果。

void ifPresent(Consumer<? super T> consumer)

  判断是否存在值,如果值存在则使用函数式编程的方式执行 consumer;否则,不做任何事情。如果没有该方法,需要先通过 isPresent() 方法对 Optional 对象进行判空再执行相应的代码:

Optional<String> opt = Optional.of("楼兰胡杨");
opt.ifPresent(str -> System.out.println(str.length()));

T get()

  get方法用于获取封装在 Optional 对象中的、不为null的值;若没有,则抛出NoSuchElementException

T orElse(T other)

  orElse(T other) 方法用于获取封装在 Optional 对象中的、不等于null的值,如果调用的对象包含null值,则返回默认值other。

    public static void main(String[] args) {
Integer num = null;
Integer orElse = Optional.ofNullable(num).orElse(12);
System.out.println("orElse null: " + orElse);
num = 32;
orElse = Optional.ofNullable(num).orElse(12);
System.out.println("orElse 32: " + orElse);
}

执行结果:

orElse null: 12
orElse 32: 32

T orElseGet(Supplier<? extends T> other)

  如果对象包含不为null的值,则返回该值;否则,使用函数式编程的方式触发 other 函数,并返回 other 执行的结果。它与 orElse(T other) 方法类似,但参数类型不同。

  Optional<Employee> op = Optional.ofNullable(null);
  // 参数是函数式接口,可以写自定义业务逻辑
  Employee emp = op.orElseGet(() -> new Employee());

Optional filter(Predicate<? super predicate)

  Optional可以实现数据过滤,与Stream流过滤功能如出一辙,也是通过匿名内部类new Predicate实现。filter() 方法的参数类型为 Predicate,也就是说可以将一个 Lambda 表达式传递给该方法作为条件,如果值非空,则根据Predicate断言条件检查该值是否符合预期,若符合则返回包含此值的Optional对象,若不符合则返回一个EMPTY的Optional对象。

public class OptionalMapFilterDemo {
public static void main(String[] args) {
String password = "password";
Optional<String> opt = Optional.ofNullable(password); Predicate<String> len6 = pwd -> pwd.length() > 6;
Predicate<String> len10 = pwd -> pwd.length() < 10;
Predicate<String> eq = pwd -> pwd.equals("password"); boolean result = opt.map(String::toLowerCase).filter(len6.and(len10 ).and(eq)).isPresent();
System.out.println(result);
// filter提供一个断言,过滤Optional的值
Optional<List<Integer>> integers = Optional.ofNullable(Arrays.asList(1, 2, 3, 4, 5)).filter(list -> list.size() > 5);
} //过滤List中null的元素
public <T> List<T> filterNullElem(Iterable<T> target) {
return ofNullable(target).filter(Objects::nonNull).collect(Collectors.toList());
}
}

  filter 与orElse或者orElseGet联合使用案例:

// 如果filter函数返回的Optional值为null,那么返回orElse方法中的值
String v1 = Optional.of("").filter(e -> e != null && e.length() > 0).orElse("default");
System.out.println(v1);
// orElseGet提供一个生成器,用于返回默认值
String v2 = Optional.of("").filter(e -> e != null && e.length() > 0).orElseGet(() -> "123");
System.out.println(v2);

Optional<U> flatMap(Function<? super T, Optional<U>> mapper)

  如果值存在,返回映射方法mapper的值;否则,返回一个空的Optional。

Optional<U> map(Function<? super T, ? extends U> mapper)

  就像stream流一样,Optional也可以通过map的方式从Optional中取出元素,map方法返回的是一个Optional类型的对象,里面的值如果为null,可以使用orElse方法赋上自定义的默认值。

  Insurance insurance = new Insurance();
insurance.setName("楼兰胡杨");
Optional<Insurance> optionalInsurance = Optional.ofNullable(insurance);
String name = optionalInsurance.map(Insurance::getName).orElse("unKnown");

  如果有值,则对其执行映射函数mapper得到返回值。如果返回值不为 null,则创建包含映射返回值的Optional作为map方法返回值;否则,返回空Optional。

T orElseThrow(Supplier<? extends X> exceptionSupplier)

  如果存在值,返回包含的值;否则,抛出由 exceptionSupplier 指定的异常:

    String v3 = Optional.of("").filter(e -> e != null && e.length() > 0).orElseThrow(RuntimeException::new);

Reference

java Stream之Optional API的更多相关文章

  1. Java Stream API性能测试

    已经对Stream API的用法鼓吹够多了,用起简洁直观,但性能到底怎么样呢?会不会有很高的性能损失?本节我们对Stream API的性能一探究竟. 为保证测试结果真实可信,我们将JVM运行在-ser ...

  2. 恕我直言你可能真的不会java第2篇:Java Stream API?

    一.什么是Java Stream API? Java Stream函数式编程接口最初是在Java 8中引入的,并且与lambda一起成为Java开发的里程碑式的功能特性,它极大的方便了开放人员处理集合 ...

  3. Java Stream 使用详解

    Stream是 Java 8新增加的类,用来补充集合类. Stream代表数据流,流中的数据元素的数量可能是有限的,也可能是无限的. Stream和其它集合类的区别在于:其它集合类主要关注与有限数量的 ...

  4. map和flatmap的区别+理解、学习与使用 Java 中的 Optional

    转自:map和flatmap的区别 对于stream,   两者的输入都是stream的每一个元素,map的输出对应一个元素,必然是一个元素(null也是要返回),flatmap是0或者多个元素(为n ...

  5. 理解、学习与使用 Java 中的 Optional

    从 Java 8 引入的一个很有趣的特性是 Optional  类.Optional 类主要解决的问题是臭名昭著的空指针异常(NullPointerException) -- 每个 Java 程序员都 ...

  6. 理解、学习与使用 JAVA 中的 OPTIONAL<转>

    从 Java 8 引入的一个很有趣的特性是 Optional  类.Optional 类主要解决的问题是臭名昭著的空指针异常(NullPointerException) —— 每个 Java 程序员都 ...

  7. 如何更好地使用Java 8的Optional

    Java 8中的Optional<T> 是一个可以包含或不可以包含非空值的容器对象,在 Stream API中很多地方也都使用到了Optional. java中非常讨厌的一点就是nullp ...

  8. java stream collector

    Java Stream API进阶篇 本文github地址 上一节介绍了部分Stream常见接口方法,理解起来并不困难,但Stream的用法不止于此,本节我们将仍然以Stream为例,介绍流的规约操作 ...

  9. 理解、学习与使用 JAVA 中的 Optional【转载】

    这是一篇转载的文章.刚学java的时候看了好久这个Optional,但一直是懵的.今天又又遇到了,重新回来再看的时候,发现并没有那么难道那个. 转载的文章再开头处写了一个对于理解Optional很关键 ...

  10. JAVA 中的Optional (臭名昭著的空指针异常(NullPointerException))

    从 Java 8 引入的一个很有趣的特性是 Optional  类.Optional 类主要解决的问题是臭名昭著的空指针异常(NullPointerException) -- 每个 Java 程序员都 ...

随机推荐

  1. 【记录】C/C++-关于I/O的坑与教训

    吐槽 每每读取字符串时,倘若稍有灵活的操作,总会遇上诡异奇怪的事情.究其原因,就是没完全理解一些基本读写函数的机制.这次做Uva227就把I/O上的问题全暴露出来了.想来还是应该记录一些经验教训. 记 ...

  2. ssh WARNING: UNPROTECTED PRIVATE KEY FILE!

    前言 在 ssh -i 指定密钥文件 登录时,出现以下报错: Permissions 0644 for 'xxxx' are too open. It is required that your pr ...

  3. PX4 仿真环境开发整理

    博客地址:https://www.cnblogs.com/zylyehuo/ (一)PX4 仿真开发 搭建仿真环境 概念介绍及环境建议 MAVROS安装(适用于ROS1.ROS2) Ubuntu安装Q ...

  4. nginx集群同步方案

    之前公司同事写过rsync加触发nginx reload脚本,适合nginx配置内容完全一致的情况. 今天写一个同步指定文件的脚本,修改完主服务器.使用scp传输到其他nginx服务器上重启NGINX ...

  5. js 小数取整

    小数取整 var num = 123.456; // 常规方法 console.log(parseInt(num)); // 123 // 双重按位非 console.log(~~num); // 1 ...

  6. 根据二叉树的前序和中序构建树,并按层次输出(C++)vector存树

    L2-006 树的遍历 #include <bits/stdc++.h> #define int long long using namespace std; #define endl ' ...

  7. emmy断点调试

    package.cpath = package.cpath .. ';C:/Users/Administrator/AppData/Roaming/JetBrains/IntelliJIdea2021 ...

  8. docker容器运行,交互式与守护式的区别

    一.使用交互式运行容器,容器运行后直接进入到容器内部,退出容器内部后,容器直接关闭

  9. 🎀avif转png在线工具推荐

    简介 本文为avif格式图片转png图片在线工具推荐 工具 https://convertio.co/zh/avif-png/ 使用 上传avif图片 选择转换的格式 点击转换 下载 结束

  10. Robot Framework使用循环

    1.普通的For循环 在一个普通的For循环中,循环开始的关键字是 :FOR ,其中的:用于与一般关键字做区分,对于循环结 构体内的每一行,使用 \ 作为改行的行首关键字.对于循环中的变量,可以在 I ...