告别丑陋判空,一个Optional类搞定

语言: CN / TW / HK

theme: cyanosis

持续创作,加速成长!这是我参与「掘金日新计划 · 10 月更文挑战」的第3天,点击查看活动详情

概述

最近项目组内做code review,充斥着大量的、原始的、丑陋的判空语句,大致类似下面的代码:

if (user != null) { Address address = user.getAddress(); if (address != null) { Country country = address.getCountry(); if (country != null) { String isocode = country.getIsocode(); if (isocode != null) { isocode = isocode.toUpperCase(); } } } }

让整体的代码显得十分的臃肿庞大丑陋,那么怎么办呢?其实一行代码就可以搞定:

String result = Optional.ofNullable(user) .map(u -> u.getAddress()) .map(a -> a.getCountry()) .map(c -> c.getIsocode()) .orElse("default");

利用Optional这个jdk8中引入的类就可以优雅的处理,现在我们来详细讲解下这个类的使用和源码。

创建Optional实例

Optional类, 是对value值进行了包装,它的值可能是null, 也可能不是null,一共有两个方法创建Optional实例

  1. static Optional of(T value)

说明: 传入的value为null,会抛错,也就是说value不许不为null

演示:

``` @Test public void testOf() { Integer value = 2; // 正常 Optional op = Optional.of(value);

value = null;
// 报空指针
op = Optional.of(value);

} ```

源码:

  1. static Optional ofNullable(T value)

说明: 传入的value可能是null, 也可能不是null

演示:

@Test public void testOfNullable() { Integer value = 2; // 正常 Optional<Integer> op = Optional.ofNullable(value); value = null; // 不报错 op = Optional.ofNullable(value); }

源码:

获取Optional中的值

以前创建了Optional实例,现在你可能要获取里面的value, 有下面几个方法。

  1. T get()

说明: 最朴素的获取原生value的方法,如果value是空,则直接抛出异常,否则返回。

演示:

``` @Test public void testGet() { Integer value = 2; // 正常 Optional op = Optional.ofNullable(value); Integer opVal = op.get(); Assert.assertEquals(opVal, value);

op = Optional.ofNullable(null);
// 会抛出异常
op.get();

} ```

源码:

  1. T orElse(T other)

说明: 如果值存在返回,否则返回orElse中传入的other

演示:

``` @Test public void testOrElse() { // 正常 Optional op = Optional.ofNullable(2); Integer opVal = op.orElse(3); Assert.assertEquals(opVal, new Integer(2));

op = Optional.ofNullable(null);
// 为空,则返回3
opVal = op.orElse(3);
Assert.assertEquals(opVal, new Integer(3));

} ```

源码:

  1. T orElseGet(Supplier<? extends T> other)

说明: 如果存在则返回该值,否则调用other这个函数编程并返回该调用的结果。

演示:

``` @Test public void testOrElseGet() { // 正常 Optional op = Optional.ofNullable(2); Integer opVal = op.orElseGet(()-> { return new Integer(3); }); Assert.assertEquals(opVal, new Integer(2));

op = Optional.ofNullable(null);
// 为空,则返回3
opVal = op.orElseGet(()-> {
    return new Integer(3);
});
Assert.assertEquals(opVal, new Integer(3));

} ```

源码:

问题: orElseGet和orElse有什么区别吗?

orElse() 和 orElseGet() 的不同之处在于当 ofNullable() 传入参数不为空时,orElse() 方法仍然创建了 other这个 对象。与之相反,orElseGet() 方法不创建对象。在执行较密集的调用时,比如调用 Web 服务或数据查询,这个差异会对性能产生重大影响。

而且我们还可以在orElseGet方法中加些日志,可以把这种为空的异常情况暴露出来。

  1. T orElseThrow(Supplier<? extends X> exceptionSupplier)

说明: 如果存在则返回该值,否则为空的话可以抛出自定义的异常。

演示:

@Test public void testOrElseThrow() { Optional<Integer> op = Optional.ofNullable(null); // 为空,则抛出指定的异常类型 Integer opVal = op.orElseThrow(()-> { return new RuntimeException(); }); // 或抛出runtime异常 Assert.assertEquals(opVal, new Integer(3)); }

源码:

判断Optional是否为空

  1. boolean isPresent()

说明: 判断value是否为空

演示:

@Test public void testIsPresent() { Optional<Integer> op = Optional.ofNullable(null); // 为空 Assert.assertFalse(op.isPresent()); }

源码:

  1. void ifPresent(Consumer<? super T> consumer)

说明: 如果存在值,则调用对应的consumer方法,否则不执行任何操作。

演示:

@Test public void testIfPresent() { Optional<Integer> op = Optional.ofNullable(5); op.ifPresent(value -> { // 如果存在,打印出来 System.out.println(value); }); }

源码:

Optional中的过滤、转换方法

此外Optional额外还提供了一些过滤、转换的方法。

  1. Optional filter(Predicate<? super T> predicate)

说明: 对Optional中的value进行过滤,如果不匹配,返回空

演示:

@Test public void testFilter() { // 不满足过滤条件,返回空 Optional Optional<String> op = Optional.ofNullable("10").filter(item -> "15".equals(item)); Assert.assertFalse(op.isPresent()); }

源码:

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

说明: 对Optional中的value进行转换映射为另外一个对象,如果value为空,返回empty Optional

演示:

源码:

小结: 回到概述的案例,就是通过不断的map, 链式调用返回内层对象的值。

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

说明: 接受一个返回值为Optional的映射函数参数,该返回值亦是flatMap方法的返回值若结果为空,则返回 空Optional。它也map的区别,我们用一个例子演示出来。

演示: 源码:

小结: 如果对于返回值非Optional类型,可以用map方法, 否则使用flatMap更加方便

总结

本文讲解了Optional这个类的使用和源码,非常的简单,实用,大家可以在自己的项目中用起来了。

参考

https://juejin.cn/post/7142813551210528776