user.getAddress().getProvince();
当user为null时,可能报NullPointerException异常的。
加上判空方法:
if(user!=null){
Address address = user.getAddress();
if(address!=null){
String province = address.getProvince();
}
}
这种写法比较丑陋,JAVA8提供了Optional类来优化这种写法
API:
1、optional、ofNullable Optional(T value), empty(), of(T value), ofNullable(T value)Optional(T value),即构造函数,它是private权限的,不能由外部调用的。
其余三个函数是public权限,供我们所调用。那么,Optional的本质,就是内部储存了一个真实的值,在构造的时候,就直接判断其值是否为空。直接上Optional(T value)构造函数的源码,如下图所示
Optional(T value)//**of(T value)**的源码 public staticOptional of(T value) { return new Optional<>(value); //of(T value)函数内部调用了构造函数 }
-
通过of(T value)函数所构造出的Optional对象,当Value值为空时,依然会报NullPointerException。
-
通过of(T value)函数所构造出的Optional对象,当Value值不为空时,能正常构造Optional对象。
除此之外呢,Optional类内部还维护一个value为null的对象,大概就是长下面这样的
public final class OptionalofNullable(T value)的作用{ //省略.... private static final Optional> EMPTY = new Optional<>(); private Optional() { this.value = null; } //empty()的作用就是返回EMPTY对象。 public static Optional empty() { @SuppressWarnings("unchecked") Optional t = (Optional ) EMPTY; return t; } }
public staticOptional ofNullable(T value) { return value == null ? empty() : of(value); }
相比较of(T value)的区别就是,当value值为null时,of(T value)会报NullPointerException异常;ofNullable(T value)不会throw Exception,ofNullable(T value)直接返回一个EMPTY对象。
是不是意味着只用ofNullable函数而不用of函数呢?
当我们在运行过程中,不想隐藏NullPointerException。而是要立即报告,这种情况下就用Of函数。
2、orElse、orElseGet、orElseThrow orElse(T other),orElseGet(Supplier extends T> other)和orElseThrow(Supplier extends X> exceptionSupplier) orElse和orElseGet的用法如下所示,相当于value值为null时,给予一个默认值:@Test
public void test() {
User user = null;
user = Optional.ofNullable(user).orElse(createUser());
user = Optional.ofNullable(user).orElseGet(() -> createUser());
}
public User createUser(){
User user = new User();
user.setName("zhangsan");
return user;
}
这两个函数的区别:
当user值不为null时,orElse函数依然会执行createUser()方法;
而orElseGet函数并不会执行createUser()方法,大家可自行测试。
至于orElseThrow,就是value值为null时,直接抛一个异常出去,用法如下所示
User user = null;
Optional.ofNullable(user).orElseThrow(()->new Exception("用户不存在"));
3、map、flatmap
map(Function super T, ? extends U> mapper)和flatMap(Function super T, Optional> mapper)
这两个函数放在一组记忆,这两个函数做的是转换值的操作。
源码:
public final class Optional{ //省略.... public Optional map(Function super T, ? extends U> mapper) { Objects.requireNonNull(mapper); if (!isPresent()) return empty(); else { return Optional.ofNullable(mapper.apply(value)); } } //省略... public Optional flatMap(Function super T, Optional> mapper) { Objects.requireNonNull(mapper); if (!isPresent()) return empty(); else { return Objects.requireNonNull(mapper.apply(value)); } } }
这两个函数,在函数体上没什么区别。唯一区别的就是入参,map函数所接受的入参类型为Function super T, ? extends U>,而flapMap的入参类型为Function super T, Optional>。
4、isPresent()和ifPresent(Consumer super T> consumer)isPresent即判断value值是否为空,而ifPresent就是在value值不为空时,做一些操作。
例子://以前写法
public String getCity(User user) throws Exception{
if(user!=null){
if(user.getAddress()!=null){
Address address = user.getAddress();
if(address.getCity()!=null){
return address.getCity();
}
}
}
throw new Excpetion("取值错误");
}
//JAVA8写法
public String getCity(User user) throws Exception{
return Optional.ofNullable(user)
.map(u-> u.getAddress())
.map(a->a.getCity())
.orElseThrow(()->new Exception("取指错误"));
}
转自:zjhred
链接:https://sourl.cn/m9CXiF



