Optional 类主要解决的问题是臭名昭著的空指针异常(NullPointerException)
Optional 的三种构造方式:
Optional.of(obj), Optional.ofNullable(obj) 和明确的 Optional.empty()
Optional.of(obj): 它要求传入的 obj 不能是 null 值的, 否则还没开始进入角色就倒在了 NullPointerException 异常上了.
Optional.ofNullable(obj): 它以一种智能的, 宽容的方式来构造一个 Optional 实例. 来者不拒, 传 null 进到就得到 Optional.empty(), 非 null 就调用 Optional.of(obj).
orElse()
它的工作方式非常直接,如果有值则返回该值,否则返回传递给它的参数值
String str = null;
String str2 = "11";
// result结果为22
String result = Optional.ofNullable(str).orElse("22");
//result2结果为11
String result2 = Optional.ofNullable(str2).orElse("22");
orElseGet()
这个方法会在有值的时候返回值,如果没有值,它会执行作为参数传入的 Supplier(供应者) 函数式接口,并将返回其执行结果
orElse() 和 orElseGet() 的不同之处
乍一看,这两种方法似乎起着同样的作用。然而事实并非如此。我们创建一些示例来突出二者行为上的异同。
我们先来看看对象为空时他们的行为:
User user = null;
//User user2 = new User("john@gmail.com", "1234");
System.out.println("orElse ");
User result = Optional.ofNullable(user).orElse(createNewUser());
System.out.println("orElseGet");
User result2 = Optional.ofNullable(user).orElseGet(() -> createNewUser());
private User createNewUser() {
System.out.println("create new user");
return new User("extra@gmail.com", "1234");
}
结果如下:
我们接下来看一个类似的示例,但这里 Optional 不为空:
User user2 = new User("john@gmail.com", "1234");
System.out.println("orElse ");
User result = Optional.ofNullable(user2).orElse(createNewUser());
System.out.println("orElseGet");
User result2 = Optional.ofNullable(user2).orElseGet(() -> createNewUser());
这个示例中,两个 Optional 对象都包含非空值,两个方法都会返回对应的非空值。不过,orElse() 方法仍然创建了 User 对象。与之相反,orElseGet() 方法不创建 User 对象。
orElseThrow()
它会在对象为空的时候抛出异常,而不是返回备选的值
这个方法让我们有更丰富的语义,可以决定抛出什么样的异常,而不总是抛出 NullPointerException。
Optional<String> str3 = Optional.ofNullable(null);
String result3 = str3.orElseThrow(()->new DataDoesNotExistException("获取数据失败"));