美文网首页
Optional 使用

Optional 使用

作者: 小丸子的呆地 | 来源:发表于2021-12-11 19:15 被阅读0次

    API

    1. empty()

    返回一个Optional容器对象,而不是 null。

    2. of(T value)

    创建一个Optional对象,如果 value 是 null,则抛出 NPE。

    3. ofNullable(T value)

    同上,创建一个Optional对象,但 value 为空时返回Optional.empty()

    4. get()

    返回Optional中包装的值,在判空之前,千万不要直接使用!

    5. orElse(T other)

    同样是返回Optional中包装的值,但不同的是当取不到值时,返回你指定的 default。

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

    同样是返回Optional中包装的值,取不到值时,返回你指定的 default。

    7. orElseThrow(Supplier<? extends X> exceptionSupplier)

    返回Optional中包装的值,取不到值时抛出指定的异常。

    8. isPresent()

    判断Optional中是否有值,返回 boolean,某些情况下很有用,但尽量不要用在 if 判断体中。

    9. ifPresent(Consumer<? super T> consumer)

    判断Optional中是否有值,有值则执行 consumer,否则什么都不干。

    10. filter(Predicate<? super T> predicate)

    过滤Optional中的值,断言通过返回原值,否则返回empty()。

    11. map(Function<? super T, ? extend U> mapper)

    转换Optional中的值。

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

    转换Optional中的值,与map的区别是,flatMap不会自动包装结果,需要mapper返回Optional结果。

    最佳实践

    首先是一些基本原则:

    • 不要声明任何Optional实例属性

    • 不要在任何 setter 或者构造方法中使用Optional

    • Optional属于返回类型,在业务返回值或者远程调用中使用

    1. 业务上需要空值时,不要直接返回 null,使用**Optional.empty()**

    public Optional<User> getUser(String name) {
        if (StringUtil.isNotEmpty(name)) {
            return RemoteService.getUser(name);
        } 
        return Optional.empty();
    }
    

    2. 使用 orElseGet()

    获取 value 有三种方式:get() orElse() orElseGet()。这里推荐在需要用到的地方只用 orElseGet()

    首先,get()不能直接使用,需要结合判空使用。这和!=null其实没多大区别,只是在表达和抽象上有所改善。

    其次,为什么不推荐orElse()呢?因为orElse()无论如何都会执行括号中的内容, orElseGet()只在主体 value 是空时执行,下面看个例子:

    public String getName() {
        System.out.print("method called");
    }
    
    String name1 = Optional.of("String").orElse(getName()); //output: method called
    String name2 = Optional.of("String").orElseGet(() -> getName()); //output:
    

    如果上面的例子getName()方法是一个远程调用,或者涉及大量的文件 IO,代价可想而知。

    orElse()就一无是处吗?并不是。orElseGet()需要构建一个Supplier,如果只是简单的返回一个静态资源、字符串等等,直接返回静态资源即可。

    public static final String USER_STATUS = "UNKNOWN";
    ...
    public String findUserStatus(long id) {
        Optional<String> status = ... ; // 
        return status.orElse(USER_STATUS);
    }
    
    //不要这么写
    public String findUserStatus(long id) {
        Optional<String> status = ... ; // 
        return status.orElse("UNKNOWN");//这样每次都会新建一个String对象
    }
    

    3. 使用 orElseThrow()

    这个针对阻塞性的业务场景比较合适,例如没有从上游获取到用户信息,下面的所有操作都无法进行,那此时就应该抛出异常。正常的写法是先判空,再手动 throw 异常,现在可以集成为一行:

    public String findUser(long id) {
        Optional<User> user = remoteService.getUserById(id) ;
        return user.orElseThrow(IllegalStateException::new);
    }
    

    4. 不为空则执行时,使用 ifPresent()

    这点没有性能上的优势,但可以使代码更简洁:

    if (status.isPresent()) {
        System.out.println("Status: " + status.get());
    }
    
    //现在
    status.ifPresent(System.out::println);
    

    5. 不要滥用

    有些简单明了的方法,完全没必要增加Optional来增加复杂性。

    public String fetchStatus() {
        String status = getStatus() ;
        return Optional.ofNullable(status).orElse("PENDING");
    }
    
    //判断一个简单的状态而已
    public String fetchStatus() {
        String status = ... ;
        return status == null ? "PENDING" : status;
    }
    

    首先,null 可以作为集合的元素之一,它并不是非法的;其次,集合类型本身已经具备了完整的空表达,再去包装一层Optional也是徒增复杂,收益甚微。例如,map 已经有了getOrDefault()这样的类似orElse()的 API 了。

    总结

    Optional的出现使 Java 对 null 的表达能力更近了一步,好马配好鞍,合理使用可以避免大量的 NPE,节省大量的人力物力。以上内容也是本人查询了很多资料,边学边写的产出,如有错漏之处,还请不吝指教。

    相关文章

      网友评论

          本文标题:Optional 使用

          本文链接:https://www.haomeiwen.com/subject/arnkfrtx.html