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

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

概述


最近项目组内做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<Integer> op = Optional.of(value);
    value = null;
    // 报空指针
    op = Optional.of(value);
}

源码:

1671190697557.jpg

  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);
    }

源码:

1671190772372.jpg


获取Optional中的值


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

  1. T get()

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

演示:

@Test
public void testGet() {
    Integer value = 2;
    // 正常
    Optional<Integer> op = Optional.ofNullable(value);
    Integer opVal = op.get();
    Assert.assertEquals(opVal, value);
    op = Optional.ofNullable(null);
    // 会抛出异常
    op.get();
}

源码:

1671190785008.jpg

  1. T orElse(T other)

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

演示:

@Test
public void testOrElse() {
    // 正常
    Optional<Integer> 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));
}

源码:

1671190809277.jpg

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

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

演示:

@Test
public void testOrElseGet() {
    // 正常
    Optional<Integer> 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));
}

源码:

1671190820984.jpg

问题: 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));
}

源码:

1671190844899.jpg


判断Optional是否为空


  1. boolean isPresent()

说明: 判断value是否为空

演示:

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

源码:

1671190862587.jpg

  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);
    });
}

源码:

1671190877429.jpg


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());
}

源码:

1671190895999.jpg

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

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

演示:

1671190902844.jpg

源码:

1671190908884.jpg

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

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

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

演示:1671190915832.jpg源码:

1671190922838.jpg

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


总结


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

目录
相关文章
|
8月前
|
JavaScript Dubbo Java
还用 if(obj!=null) 做非空判断?带你快速上手 Optional 实战性理解
1.前言 2.认识Optional并使用 3.实战场景再现 4.Optional使用注意事项 5.jdk1.9对Optional优化
|
8月前
|
人工智能 前端开发 JavaScript
【炫技的代码写法】
【炫技的代码写法】
|
1月前
|
安全 Java 程序员
Dating Java8系列之巧用Optional之优雅规避NPE问题
Dating Java8系列之巧用Optional之优雅规避NPE问题
33 0
|
Java 测试技术
开发小技巧系列 - 如何避免NPE,巧用Optional重构三元表达式?(三)
NPE是一个老生长谈的问题,无论新手,还是老手,在开发程序的过程中,都不可避免会遇到,而为了处理NPE,往往需要添加很多重复性的检查代码,又长又臭。NPE系列文章,是总结了过往的开发经验,助力更多新手,避免踩坑。
74 0
|
Java 测试技术 API
开发小技巧系列 - 如何避免NPE,去掉if...else(四)
利用optional来处理各种IF-ELSE的判断
89 0
|
NoSQL Java API
RedisTemplate 接口误用造成的空指针异常记录(深扒multiGet接口)
RedisTemplate 接口误用造成的空指针异常记录(深扒multiGet接口)
247 0
|
消息中间件 JavaScript 小程序
告别丑陋判空,一个Optional类就能搞定!
告别丑陋判空,一个Optional类就能搞定!
|
JSON Java fastjson
阿粉带你彻底解决开发中对象Bean与Map互转问题!
在实际开发过程中,经常碰到需要进行对象与map之间互转的问题,其实对于对象、Map 之间进行互转有很多种方式,下面我们一起来梳理一下:
1122 0
阿粉带你彻底解决开发中对象Bean与Map互转问题!
|
消息中间件 JavaScript 小程序
Lombok 同时使用 @Data 和 @Builder 的巨坑,千万别乱用!
Lombok 同时使用 @Data 和 @Builder 的巨坑,千万别乱用!
|
JSON API 数据格式
利用注解 + 反射消除重复代码,妙
利用注解 + 反射消除重复代码,妙
86 0