Skip to content

Optional类 🚀

java中尝试访问空引用的属性调用空引用的方法是会报空指针NullPointerException异常。实际项目中会处理大量为空的值,代码会有很多的条件判断,难以阅读与维护。

java
if(user!=null) {
	System.out.println(user.getFullName());
} else {
	User defaultUser = new User("Stark", "Tony Stark");
	System.out.println(defaultUser.getFullName());
}

Optional是什么? 🚀

Optional类引入了一种显式的方式来处理可能为空的对象,强制程序员在可能为空的情况下进行显式的处理,以避免空指针异常。Optional类似容器,可以包含各种类型的值,也可以为null。Optional类提供了一系列方法来方便地操作内部的值。常用的方法有get、orElse、orElseGet、orElseThrow等。

Optional的设计也考虑了函数式编程的原则,可以与Lambda表达式和StreamAPI等特性结合使用,可以进行链式调用替代命令式编程的方式通过编写if条件语句检查null值。

Optional方法 🚀

首先我们需要创建两个类,User类与UserRepository类。

java
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;

@Data
@NoArgsConstructor
@AllArgsConstructor
public class User {
    String name;
    String fullName;
}
java
import java.util.Optional;

public class UserRepository {
    public User findUserByName(String name){
        if(name.equals("Peter")){
            return new User("Peter","Peter Parker");
        } else {
            return null;
        }
    }
}

isPresent、isEmpty方法

isPresent方法用于检查optional内是否存在值,返回为布尔值,存在为true,不存在为false。

isEmpty方法用于检查optional内是否为空,返回为布尔值,为空为true,不为空为false。

java
import lombok.extern.slf4j.Slf4j;
import org.junit.Test;
import java.util.Optional;

@Slf4j
public class MyTest {
    @Test
    public void test() {
        // 创建值为null的Optional对象
        Optional<Object> optionalBox = Optional.empty();
        System.out.println(optionalBox.isPresent());
        System.out.println(optionalBox.isEmpty());
    }
}

输出

false
true

empty、of、ofNullable方法

若创建的对象为null,则可以采用empty()方法。

java
	Optional<Object> optionalBox = Optional.empty();

创建不为null的对象需要调用of方法,此时若value为null,则抛出NullPointerException异常。

java
    String value = "Peter";
    Optional<String> optionalBox = Optional.of(value);

若想要创建可能为null的对象,可以使用ofNullable( )方法。

java
    String value = "Peter";
    Optional<String> optionalBox = Optional.ofNullable(value);

get、orElse、orElseGet与orElseThrow方法

取值可以使用get()方法,若取值对象为null,会报java.util.NoSuchElementException异常(并非java.lang.NullPointerException异常)。

java
    String value2 = optionalBox.get();
    System.out.println(value2);

我们这里修改UserRepository类中的findUserByName方法,使其返回值为Optional对象。

java
public Optional<User> findUserByName(String name){
    if(name.equals("Peter")) {
        return Optional.of(new User("Peter","Peter Parker"));
    } else {
        return Optional.empty();
    }
}

orElse是Optional类中的一个重要方法,它用于获取值或在值为空的情况下提供一个默认值。

java
    UserRepository userRepository = new UserRepository();
    Optional<User> optionalUser = userRepository.findUserByName("Peter2");
    User user = optionalUser.orElse(new User("Stark","Tony Stark"));
    System.out.println(user.getFullName());

orElse方法即使读到的数据不为null,仍会创建一个新User对象。因此最好使用只有读到的数据为null的时候才会新建对象的orElseGet方法。

orElseGet方法的参数为Supplier的函数式接口,需要使用Lambda表达式实现。

java
    optionalUser.orElseGet(()->new User("Stark","Tony Stark"));
    System.out.println(user.getFullName());

orElseThrow方法用于在 Optional 对象中的值为空时抛出一个指定的异常。

orElseThrow方法通过Supplier的函数式接口,可以生成自定义异常,默认抛出异常为java.util.NoSuchElementException

java
    UserRepository userRepository = new UserRepository();
    Optional<User> optionalUser = userRepository.findUserByName("Peter2");
    optionalUser.orElseThrow(()->new RuntimeException("User not found"));

ifPresent、ifPresentOrElse、filter方法

ifPresent方法参数中若对象不为null,则会执行Labmda中的方法;

若参数对象为null,则不会执行Labmda中的方法,也不会报错。

java
    UserRepository userRepository = new UserRepository();
    Optional<User> optionalUser = userRepository.findUserByName("Peter2");
    optionalUser.ifPresent(user -> System.out.println(user.getFullName()));

当我们希望值为空时进行其他操作,需要使用ifPresentOrElse方法。

java
    optionalUser.ifPresentOrElse(user -> System.out.println(user.getFullName()),
            ()->System.out.println("User not found"));

若满足filter方法中的条件,则会返回包含值的Optional对象,如果不满足,则返回空的Optional对象

java
    Optional<User> optionalUser2 = optionalUser.filter(user -> user.getFullName().equals("Peter Parker"));
    System.out.println(optionalUser2.isPresent());

Map与flatMap方法

进行两个方法前需要将User类中的 getFullName方法返回值修改为Optional。

java
public Optional<String> getFullName(){
    return Optional.ofNullable(this.fullName);
}

map方法:对Optional中的值进行转换(若值为空,则map方法什么也不会做,直接返回空的Optional对象)。这个变换基于提供该map的函数,并且这个变换是可选的,如果optional的值为空则不会做任何改变,并且map方法不会改变原始的Optional对象,而返回新的Optional对象,因此可以链式调用进行多个转换操作。

java
    UserRepository userRepository = new UserRepository();
    Optional<User> optionalUser = userRepository.findUserByName("Peter");
    Optional<String> optionalFullName = optionalUser.map(User::getFullName);
    System.out.println(optionalFullName.get());

flatmap方法用于扁平化嵌套的Optional结构,以避免引入不必要的嵌套层级,具体为flatmap的转换函数返回的必须是另一个Optional对象,意味着flatMap方法可以用于嵌套的Optional情况,可以将两个为嵌套关系的Optional对象转换为一个。如果原始的Optional对象为空,或转换函数返回的Optional对象为空,那么最终得到的也是为空的Optional对象。

java
    UserRepository userRepository = new UserRepository();
    Optional<User> optionalUser = userRepository.findUserByName("Peter");
    Optional<String> optional = optionalUser.flatMap(User::getFullName);

若只需要对Optional对象中的值进行转换,而不需要嵌套的Optional,那么使用map方法更合适。如果要进行一些操作返回另外一个Optional对象,flatmap方法更合适。

Stream方法

Optional的stream方法,可以将Optional对象转换为Stream对象,对其中的值进行流操作,如果Optional对象包含值,则将这个值封装到一个Stream流中,如果Optional对象为空,则创造一个为空的Stream流。

java
    UserRepository userRepository = new UserRepository();
    Optional<User> optionalUser = userRepository.findUserByName("Peter");
    Stream<String> a =optionalUser.map(User::getName).stream();
    a.forEach(System.out::println);

Optional不适用场景 🚀

  • 不应该用于类的字段,会增加内存消耗,并使序列化变得复杂;
  • 不应该用于方法参数,使方法的理解和使用变得复杂;
  • 不应用于构造器参数,迫使调用者创建Optional实例,应该通过构造器重载解决;
  • 不应该用于集合的参数,集合已经很好的处理空集合的情况,没必要使用Optional包装集合;
  • 不建议使用get方法,若为null会报错。

更新时间: