Java8以前咱们在写代码的时候,常常会遇到返回null的状况,若是这种状况不加以判断,你就会碰到NullPointerException(NPE)。而在Java8中,Optional类型是一种更好的表示缺乏返回值的形式。java
首先来看一段代码,这多是之前大多数人的写法web
private void getIsoCode( User user){
if (user != null) {
Address address = user.getAddress();
if (address != null) {
Country country = address.getCountry();
if (country != null) {
String isocode = country.getIsocode();
if (isocode != null) {
isocode = isocode.toUpperCase();
}
}
}
}
}
复制代码
而当咱们有了Optional以后,上面的代码就能够缩减很大一部分,答案我会在后面给出。数据库
我见过有人的写法是这样的安全
Optional<T> optionalValue = ... ;
optionalValue.get().someMethod();
或者
if(optionalValue.isPresent()) {
optionalValue.get().someMethod();
}
复制代码
它并不比下面的方式安全app
T value = ... ;
value.someMethod();
或者
if(value != null) {
value.someMethod();
}
复制代码
若是你在你的代码中出现了上面使用Optional的片断,那么你该好好优化下了。函数
其实高效使用Optional的关键在于,使用一个 接受正确值或者返回另外一个替代值 的方法。性能
// 建立一个空的Optional实例
public static<T> Optional<T> empty() {
Optional<T> t = (Optional<T>) EMPTY;
return t;
}
// 根据传入的值建立一个非空的实例, value不能为空,不然抛出NPE
public static <T> Optional<T> of(T value) {
return new Optional<>(value);
}
// 根据传入的值建立实例,value能够为空
public static <T> Optional<T> ofNullable(T value) {
return value == null ? empty() : of(value);
}
复制代码
上面的三个方法就是咱们构造Optional的途径,能够看到实际上Optional是对value的封装。
须要注意的是ofNullable
和 of
的区别,推荐使用ofNullable
方法优化
boolean isPresent()
判断value是否存在spa
Optional filter(Predicate predicate)
判断value是否知足条件code
Optional<U> map(Function<? super T, ? extends U> mapper)
对其中的value执行一个函数,将其变成另外一个值。返回的值会被Optional.ofNullable封装
Optional<U> flatMap(Function<? super T, Optional<U>> mapper)
也是对其中的value执行一个函数,注意和map的区别在于,执行这个函数返回的值是Optional类型,返回的值不会被封装。
orElse(T other)
当value存在时,返回value,不存在时返回other
orElseGet(Supplier<? extends T> other)
和orElse同样,只是这里的参数是经过传入的function来决定的
T orElseThrow
当value存在时,返回value,不存在时抛出异常
开头给出的代码就能够被优化为下面的代码
String isoCode = Optional.ofNullable(user)
.map(User::getAddress) //Optional<Address>
.map(Address::getCountry) //Optional<Country>
.map(Country::getIsocode) // Optional<String>
.orElse("empty");
复制代码
不过有一点须要注意的是orElse和orElseGet的区别在于,不管是否知足条件orElse中的方法始终会被执行,而orElseGet中的只有当value为空时才会执行。
Optional.ofNullable(user)
.map(User::getAddress)
.map(Address::getCountry)
.map(Country::getIsocode)
.orElse(getIsoCode());
复制代码
你可能以为没什么,可是若是你的业务中获取这个值要去数据库查询,那么每一次只要运行这个代码就都要去查询,这样就形成了没必要要的性能损失了,仍是一个很大的问题的。