一、前言
在Java中对一个空对象进行操作时,便会抛出最常见的异常NullPointerException。为了改善这个问题,Java 8中提供了一个java.util.Optional
我们县创建好测试实体:
Address.java
1 | @Data |
User.java
1 | @Data |
二、创建Optional
2.1 创建一个空的Optional
我们可以使用静态工厂方法Optional.empty,创建一个空的Optional对象:
1 | Optional<User> user = Optional.empty(); |
2.2 根据非空值创建Optional
我们也可以使用静态工厂方法Optional.of来创建一个非空对象的Optional对象:
1 | Optional<User> user = Optional.of(user); |
如果user为空,这段代码会立即抛出一个NullPointerException。
2.3 创建可以为null的Optional
使用静态工厂方法Optional.ofNullable,我们可以创建一个允许null值的Optional对象:
1 | Optional<User> user = Optional.ofNullable(user); |
如果user为空,对其调用get方法将抛出NoSuchElementException。
三、Optional方法
3.1 Optional(T value),empty(),of(T value),ofNullable(T value)
这四个函数之间具有相关性,因此放在一组进行记忆。
先说明一下,Optional(T value),即构造函数,它是private权限的,不能由外部调用的。其余三个函数是public权限,供我们所调用。那么,Optional的本质,就是内部储存了一个真实的值,在构造的时候,就直接判断其值是否为空。
那么,**of(T value)**的源码如下
1 | public static <T> Optional<T> of(T value) { |
也就是说of(T value)函数内部调用了构造函数。根据构造函数的源码我们可以得出两个结论:
通过of(T value)函数所构造出的Optional对象,当Value值为空时,依然会报NullPointerException。
通过of(T value)函数所构造出的Optional对象,当Value值不为空时,能正常构造Optional对象。
除此之外呢,Optional类内部还维护一个value为null的对象,大概就是长下面这样的
1 | public final class Optional<T> { |
那么,empty()的作用就是返回EMPTY对象。
好了铺垫了这么多,可以说ofNullable(T value)的作用了,上源码
1 | public static <T> Optional<T> ofNullable(T value) { |
好吧,大家应该都看得懂什么意思了。相比较of(T value)的区别就是,当value值为null时,of(T value)会报NullPointerException异常;ofNullable(T value)不会throw Exception,ofNullable(T value)直接返回一个EMPTY对象。
那是不是意味着,我们在项目中只用ofNullable函数而不用of函数呢?
不是的,一个东西存在那么自然有存在的价值。当我们在运行过程中,不想隐藏NullPointerException。而是要立即报告,这种情况下就用Of函数。但是不得不承认,这样的场景真的很少。博主也仅在写junit测试用例中用到过此函数。
3.2 orElse(T other),orElseGet(Supplier<? extends T> other)和orElseThrow(Supplier<? extends X> exceptionSupplier)
这三个函数放一组进行记忆,都是在构造函数传入的value值为null时,进行调用的。orElse和orElseGet的用法如下所示,相当于value值为null时,给予一个默认值:
1 | @Test |
这两个函数的区别:当user值不为null时,orElse函数依然会执行createUser()方法,而orElseGet函数并不会执行createUser()方法,大家可自行测试。
至于orElseThrow,就是value值为null时,直接抛一个异常出去,用法如下所示
1 | User user = null; |
3.3 map(Function<? super T, ? extends U> mapper)和flatMap(Function<? super T, Optional> mapper)
这两个函数放在一组记忆,这两个函数做的是转换值的操作。
1 | public final class Optional<T> { |
这两个函数,在函数体上没什么区别。唯一区别的就是入参,map函数所接受的入参类型为Function<? super T, ? extends U>,而flapMap的入参类型为Function<? super T, Optional>。
在具体用法上,对于map而言:
如果User结构是下面这样的
1 | public class User { |
这时候取name的写法如下所示
1 | String city = Optional.ofNullable(user).map(u-> u.getName()).get(); |
对于flatMap而言:
如果User结构是下面这样的
1 | public class User { |
这时候取name的写法如下所示
1 | String city = Optional.ofNullable(user).flatMap(u-> u.getName()).get(); |
3.4 isPresent()和ifPresent(Consumer<? super T> consumer)
这两个函数放在一起记忆,isPresent即判断value值是否为空,而ifPresent就是在value值不为空时,做一些操作。这两个函数的源码如下
1 | public final class Optional<T> { |
需要额外说明的是,大家千万不要把
1 | if (user != null){ |
给写成
1 | User user = Optional.ofNullable(user); |
因为这样写,代码结构依然丑陋。博主会在后面给出正确写法
至于ifPresent(Consumer<? super T> consumer),用法也很简单,如下所示
1 | Optional.ofNullable(user).ifPresent(u->{ |
3.5 filter(Predicate<? super T> predicate)
1 | public final class Optional<T> { |
filter 方法接受一个 Predicate 来对 Optional 中包含的值进行过滤,如果包含的值满足条件,那么还是返回这个 Optional;否则返回 Optional.empty。
1 | Optional<User> user1 = Optional.ofNullable(user).filter(u -> u.getName().length()<6); |
如上所示,如果user的name的长度是小于6的,则返回。如果是大于6的,则返回一个EMPTY对象。
四、实战使用
4.1 例1
在函数方法中
以前写法
1 | public String getCity(User user) throws Exception{ |
JAVA8写法
1 | public String getCity(User user) throws Exception{ |
4.2 例2
比如,在主程序中
以前写法
1 | if(user!=null){ |
JAVA8写法
1 | Optional.ofNullable(user) |
4.3 例3
以前写法
1 | public User getUser(User user) throws Exception{ |
JAVA8写法
1 | public User getUser(User user) { |