jdk8新特性Optional

前言

Java世界中最为常见的异常就是npe了,所以使用对象前对对象判空是保证代码鲁棒性的重要保证,因此在jdk7之前我们经常写出这样的代码:

        if (someObject != null) {
            if (someObject.someField != null) {
                if (someObject.someField.someField != null) {
                    Systemout.println(someObject.someField.someField)
                }
            }
        }

代码逻辑虽然不复杂但是过于繁复,于是在jdk8提供了一个新的特性Optional,结合lambda表达式可以极大地简化我们的判空操作,先来看看使用Optional如何重构上面那一段代码

Optional.ofNullable(someObject).map(SomeObject::getSomeField).map(SomeField::getSomeField).ifPresent(System.out::println);

可以看到借助于lambda表达式,之前的多重嵌套的判空操作只需要一行代码就可以完成。

再来看看通过Optional规避集合越界的代码

List list = new ArrayList();

Optional.ofNullable(list).filter(e -> e.size() > 0).map(e -> e.get(1)).ifPresent(System.out::println);

这里是有了Optional中的filter操作符,只有符合lambda表达式条件的元素才不会被过滤进入下一个操作符map,因此长度不符合要求的数组根本不会被取值。

除了上述的map,filter和isPresent操作符,Optional还提供了一系列其他的常用操作符。

Optional操作符

1 ofNullable

传入的对象可以为null

    public static <T> Optional<T> ofNullable(T value) {
        return value == null ? empty() : of(value);
    }

2 of

传入的对象不可以为null,否则抛出NullPointerException

    public static <T> Optional<T> of(T value) {
        return new Optional<>(value);
    }

例子

public static void main(String[] args) {

    User user = new User();
    User user1 = null;

    // 传递进去的对象不可以为null,如果为null则抛出异常
    Optional<User> op1 = Optional.of(user1);

    // 传递进去的对象可以为null,如果为null则返回一个没有装载对象的Optional容器
    Optional<User> op2 = Optional.ofNullable(user);
}

3 ifPresent

首先来看看ifPresent(Consumer<? super T> consumer)方法

public void ifPresent(Consumer<? super T> consumer) {
    if (value != null)
        consumer.accept(value);
}

@FunctionalInterface
public interface Consumer<T> {
    void accept(T t);
}

复制代码如果容器中的对象存在,则调用accept方法,比如说:

public static void main(String[] args) {

    User user = new User();
    user.setName("Java3y");
    test(user);
}

public static void test(User user) {

    Optional<User> optional = Optional.ofNullable(user);

    // 如果存在user,则打印user的name
    optional.ifPresent((value) -> System.out.println(value.getName()));

    // 旧写法
    if (user != null) {
        System.out.println(user.getName());
    }
}

4 orElseGet和orElseThrow

直接看源码:

// 如果对象存在,则直接返回,否则返回由Supplier接口的实现用来生成默认值
public T orElseGet(Supplier<? extends T> other) {
    return value != null ? value : other.get();
}


@FunctionalInterface
public interface Supplier<T> {
    T get();
}


// 如果存在,则返回。否则抛出supplier接口创建的异常
public <X extends Throwable> T orElseThrow(Supplier<? extends X> exceptionSupplier) throws X {
    if (value != null) {
        return value;
    } else {
        throw exceptionSupplier.get();
    }
}

例子:

public static void main(String[] args) {

    User user = new User();
    user.setName("Java3y");
    test(user);
}

public static void test(User user) {

    Optional<User> optional = Optional.ofNullable(user);

    // 如果存在user,则直接返回,否则创建出一个新的User对象
    User user1 = optional.orElseGet(() -> new User());
    
    // 旧写法
    if (user != null) {
        user = new User();
    }
}

总的来说跟我们上面所讲的orElse()差不多,只不过它可以通过Supplier接口的实现来生成默认值。

5 filter

// 如果容器中的对象存在,并且符合过滤条件,返回装载对象的Optional容器,否则返回一个空的Optional容器
public Optional<T> filter(Predicate<? super T> predicate) {
    Objects.requireNonNull(predicate);
    if (!isPresent())
        return this;
    else
        return predicate.test(value) ? this : empty();
}


// 接口
@FunctionalInterface
public interface Predicate<T> {

    boolean test(T t);
}

由于返回的是Optional对象,我们就可以实现链式调用了!
例子:

public static void test(User user) {

    Optional<User> optional = Optional.ofNullable(user);

    // 如果容器中的对象存在,并且符合过滤条件,返回装载对象的Optional容器,否则返回一个空的Optional容器
    optional.filter((value) -> "Java3y".equals(value.getName()));
}

6 map

直接看源码:

// 如果容器的对象存在,则对其执行调用mapping函数得到返回值。然后创建包含mapping返回值的Optional,否则返回空Optional。
public<U> Optional<U> map(Function<? super T, ? extends U> mapper) {
    Objects.requireNonNull(mapper);
    if (!isPresent())
        return empty();
    else {
        return Optional.ofNullable(mapper.apply(value));
    }
}


// 接口
@FunctionalInterface
public interface Function<T, R> {
    R apply(T t);
}

例子:

public static void test(User user) {

    Optional<User> optional = Optional.ofNullable(user);

    // 如果容器的对象存在,则对其执行调用mapping函数得到返回值。然后创建包含mapping返回值的Optional,否则返回空Optional。
    optional.map(user1 -> user1.getName()).orElse("Unknown");
}

// 上面一句代码对应着最开始的老写法:

public String tradition(User user) {
    if (user != null) {
        return user.getName();
    }else{
        return "Unknown";
    }
}

猜你喜欢

转载自blog.csdn.net/m0_37556444/article/details/84641682