一个null引用的危害。首先创建一个类Computer,如图所示:
调用如下代码:
String version = computer.getSoundcard().getUSB().getVersion();
如果计算机没有声卡,那么调用getSoundCard()方法肯定会报NullPointerException。
怎么样避免?
Java8引入了一个新类叫做java.util.Optional,这个类的设计灵感源于Haskell语言和Scala语言。这个类可以包含了一个任意值,像下面图和代码表示的那样。你可以把Optional看作是一个有可能包含了值的值,如果Optional不包含值那么它就是空的。
public class Computer {
private Optional<Soundcard> soundcard;
public Optional<Soundcard> getSoundcard() {
... }
...
}
public class Soundcard {
private Optional<USB> usb;
public Optional<USB> getUSB() {
... }
}
public class USB{
public String getVersion(){
... }
}
上述代码展现了一个计算机可能包含一个声卡。声卡也是有可能包含一个USB端口,该模型清晰的反映了一个被给定的值是可以不存在的。
但是怎么处理Optional< SoundCard>这个对象呢?很简单,Optional;类包含了一些方法来处理值是否存在的状况。和null引用相比Optional类迫使你要做值是否为空的相关处理,从而避免空指针异常。
需要说明的是Optional并不是要取代null引用。相反地,是为了让设计的API更容易被理解,当你看到一个函数的签名时,就可以判断要传递给这个函数的值是不是有可能不存在,这就促使你要打开Optional类来处理确实值的状况了
如何使用Optional改写传统的null引用检测,如下,看完后面的内容就会理解下面的代码
String name = computer.flatMap(Computer::getSoundcard)
.flatMap(Soundcard::getUSB)
.map(USB::getVersion)
.orElse("UNKNOWN");
创建一个空Optional对象
Optional<Soundcard> sc = Optional.empty();
创建一个包含非bull值的Optional
SoundCard soundcard = new Soundcard();
Optional<Soundcard> sc = Optional.of(soundcard);
如果声卡null,空指针异常会立即被抛出,这比在获取声卡属性时抛出要好
通过使用ofNullable,可以创建一个可能包含null引用的Optional对象
Optional<Soundcard> sc = Optional.ofNullable(soundcard);
如果声卡时null引用,Optional对象就是一个空的
现在已经有了Optional对象,可以调用相应的方法来处理Optional对象中的值是否存在。和进行null检测相比,我们可以使用ifPresent()方法,像下面这样:
Optional<Soundcard> soundcard = ...;
soundcard.ifPresent(System.out::println);
这样就不必再做null检测,如果Optional对象是空的,那么信息将不会打印出来
也可以使用isPresent()方法查看Optional对象是否真的存在。另外,还有一个get()方法可以返回Optional对象中包含的值。如果不存在,则会抛出一个NoSuchElementException异常,这两个方式可以如下使用,从而避免异常:
if(soundcard.isPresent()){
System.out.println(soundcard.get());
}
当遇到null时,一个常规的操作时返回一个默认值,可以使用三元表达式实现:
Soundcard soundcard = maybeSoundcard != null ? maybeSoundcard : new Soundcard("basic_sound_card");
使用Optional对象的话,可以orElse()使用重写,当Optional是空的时候orElse()可以返回一个默认值:
Soundcard soundcard = maybeSoundcard.orElse(new Soundcard("defaut"));
类似的,当Optional为空的时候也可以使用orElseThrow()抛出异常:
Soundcard soundcard =
maybeSoundCard.orElseThrow(IllegalStateException::new);
经常会调用一个对象的方法来判断它的属性。比如,可能需要检测USB端口号是否是某个特定值。为了安全起见,需要检查指向USB的值是否是null,然后再调用getVersion()方法,如下:
USB usb = ...;
if(usb != null && "3.0".equals(usb.getVersion())){
System.out.println("ok");
}
如果使用Optional的话可以使用filter函数:
Optional<USB> maybeUSB = ...;
maybeUSB.filter(usb -> "3.0".equals(usb.getVersion()).ifPresent(() -> System.out.println("ok"));
filter方法需要一个predicate对象作为参数。如果Optional中的值存在并满足predicate,那么filter函数将会返回满足条件的值;否则,会返回一个空的Optional对象
现在已经介绍了一个可以使用Optional重构代码的例子,那么我们应该如何使用安全的方式实现下面代码呢?
String version = computer.getSoundcard().getUSB().getVersion();
注意上面的代码都是从一个对象中提取另一个对象,使用map函数可以实现。在前面的文章中我们设置了Computer中包含的是一个Optional对象,Soundcard包含的是一个Optional对象,因此我们可以这么重构代码
String version = computer.map(Computer::getSoundcard)
.map(Soundcard::getUSB)
.map(USB::getVersion)
.orElse("UNKNOWN");
不幸的是,上面的代码会编译错误,那么为什么呢?computer变量是Optional类型的,所以它调用map函数是没有问题的。但是getSoundcard()方法返回的是一个Optional< Soundcard>的对象,返回的是Optional
map函数的源码实现是这样的:
public<U> Optional<U> map(Function<? super T, ? extends U> mapper) {
Objects.requireNonNull(mapper);
if (!isPresent())
return empty();
else {
return Optional.ofNullable(mapper.apply(value));
}
}
可以看出map函数还会再调用一次Optional.ofNullable(),从而导致返回Optional
Optional提供了flatMap这个函数,它的设计意图是当对Optional对象的值进行转化(就像map操作)然后一个两级Optional压缩成一个。下面的图展示了Optional对象通过调用map和flatMap进行类型转化的不同:
因此可以这样写:
String version = computer.flatMap(Computer::getSoundcard)
.flatMap(Soundcard::getUSB)
.map(USB::getVersion)
.orElse("UNKNOWN");
第一个flatMap保证了返回的是Optional而不是Optional< Optional>,第二个flatMap实现了同样的功能从而返回的是 Optional。注意第三次调用了map(),因为getVersion()返回的是一个String对象而不是一个Optional对象。
我们终于把刚开始使用的嵌套null检查的丑陋代码改写了可读性高的代码,也避免了空指针异常的出现的代码。
在这片文章中我们采用了Java 8提供的新类java.util.Optional< T>。这个类的初衷不是要取代null引用,而是帮助设计者设计出更好的API,只要读到函数的签名就可知道该函数是否接受一个可能存在也可能不存在的值。另外,Optional迫使你去打开Optional,然后处理值是否存在,这就使得你的代码避免了潜在的空指针异常。