Java8的Optional:如何干掉空指针?
Optional概述
Optional 是个容器:它可以保存类型T的value,或者仅仅保存null。Optional提供很多有用的方法,这样我们就不用显式进行空值检测,很好地解决了空指针异常处理的问题,比如可以使用isPresent()
方法判断value是否为null,使用get()
方法获取value值等等。
Optional的构造方法是私有的,实例不能new,可以使用静态方法来创建:
// 1、创建一个包装对象值为空的Optional对象
Optional<String> optStr = Optional.empty();
// 2、创建包装对象值非空的Optional对象
Optional<String> optStr1 = Optional.of("optional");
// 3、创建包装对象值允许为空的Optional对象
Optional<String> optStr2 = Optional.ofNullable(null);
Optional简单案例
看完Optional的概述,我们用一个简单的例子说明一下:
下面这段代码接收一个User对象,如果user为null,则抛出异常【这是一个非常常规的避免空指针异常的做法,如果没有这步,getName会NPE】,否则返回user的name。
public String getName1(User user) {
if (user == null) {
throw new RuntimeException("user不能为null!");
}
return user.getName();
}
如果使用Optional,应该怎么去处理呢?
public String getName2(User user) {
return Optional.ofNullable(user) // 包装user对象,如果user为null,则返回空的Optional对象
.map(User::getName)
.orElseThrow(() -> new RuntimeException("user不能为null"));// 如果有值则返回,没有则抛出异常。
}
- Optional使用静态的
ofNullable
方法将user对象进行包装,将可能为null的user对象保护起来。
public static <T> Optional<T> ofNullable(T value) {
// empty() 方法 创建一个空的Optional对象, of对象在构造Optional的时候,value如果weinull,会引发NPE
return value == null ? empty() : of(value);
}
- orElseThrow方法接收一个Supplier 对象,这里我们在lambda表达式那节已经说过了,不再赘述,感兴趣可以查看:Java8的Lambda表达式,你会不?
public <X extends Throwable> T orElseThrow(Supplier<? extends X> exceptionSupplier) throws X {
// 如果有值,直接返回值
if (value != null) {
return value;
} else {
// 抛出异常,这个异常Supplier接口定义
throw exceptionSupplier.get();
}
}
Optional的主要方法
方法 | 描述 |
---|---|
empty |
返回一个空的 Optional 实例 |
filter |
如果值存在并且满足提供的断言, 就返回包含该值的 Optional 对象;否则返回一个空的 Optional 对象 |
map |
如果值存在,就对该值执行提供的 mapping 函数调用 |
flatMap |
如果值存在,就对该值执行提供的 mapping 函数调用,返回一个 Optional 类型的值,否则就返 回一个空的 Optional 对象 |
get |
如果该值存在,将该值用 Optional 封装返回,否则抛出一个 NoSuchElementException 异常 |
ifPresent |
如果值存在,就执行使用该值的方法调用,否则什么也不做 |
isPresent |
如果值存在就返回 true,否则返回 false |
of |
将指定值用 Optional 封装之后返回,如果该值为 null,则抛出一个 NullPointerException 异常 |
ofNullable |
将指定值用 Optional 封装之后返回,如果该值为 null,则返回一个空的 Optional 对象 |
orElse |
如果有值则将其返回,否则返回一个默认值 |
orElseGet |
如果有值则将其返回,否则返回一个由指定的 Supplier 接口生成的值 |
orElseThrow |
如果有值则将其返回,否则抛出一个由指定的 Supplier 接口生成的异常 |
关于Optional的使用建议
- 尽量避免在程序中直接调用Optional对象的get()和isPresent()方法,get方法使用时存在NPE风险,需要用isPresent()方法做特殊判断,没有达到简化的目的。
- 避免使用Optional类型声明实体类的属性,因为Optional没有实现序列化接口。