Optional 由 Java8 提供。是用于防止 NullPointerException 的优雅工具。善用 Optional 可以使我们代码中很多繁琐、丑陋的设计变得十分优雅。这篇文章是建立在你对 Optional 的用法有一定了解的基础上的,如果你还不太了解 Optional,可以先去看看相关教程,或者查阅 Java 文档。
使用 Optional,我们就可以把下面这样的代码进行改写:
public static String getUserName(User u) {
if (u == null || u.name == null)
return "Unknown";
return u.name;
}
不过,千万不要改下面那样:
public static String getUserName(User u) {
Optional<User> user = Optional.ofNullable(u);
if (!user.isPresent())
return "Unknown";
return user.get().name;
}
这样改写非但不简洁,而且其操作还是和第一段代码一样。无非就是用 isPresent 方法来替代 u==null。这样的改写并不是 Optional 正确的用法,我们再来改写一次:
public static String getUserName(User u) {
return Optional.ofNullable(u)
.map(user->user.name)
.orElse("Unknown");
}
这样才是正确使用 Optional 的姿势。那么按照这种思路,我们可以安心地进行链式调用,而不是一层层判断了。看一段代码:
public static String getChampionName(Competition comp) throws IllegalArgumentException {
if (comp != null) {
CompResult result = comp.getResult();
if (result != null) {
User champion = result.getChampion();
if (champion != null) {
return champion.getName();
}
}
}
throw new IllegalArgumentException("The value of param comp isn't available.");
}
但是我们并不能开心的一路 comp.getResult().getChampion().getName() 到底,所以讲道理在 Java 中我们怎么办?
让我们看看经过 Optional 加持过后,这些代码会变成什么样子:
public static String getChampionName(Competition comp) throws IllegalArgumentException {
return Optional.ofNullable(comp)
.map(Competition::getResult) // 相当于c -> c.getResult(),下同
.map(CompResult::getChampion)
.map(User::getName)
.orElseThrow(()->new IllegalArgumentException("The value of param comp isn't available."));
}
这就很舒服了。Optional 给了我们一个真正优雅的 Java 风格的方法来解决 null 安全问题。虽然没有直接提供一个操作符写起来短,但是代码看起来依然很爽很舒服。
还有很多不错的使用姿势,比如字符串为空则不打印可以这么写:
string.ifPresent(System.out::println);
Optional 的魅力还不止于此,Optional 还有一些神奇的用法,比如 Optional 可以用来检验参数的合法性。
public void setName(String name) throws IllegalArgumentException {
this.name = Optional.ofNullable(name)
.filter(User::isNameValid)
.orElseThrow(()->new IllegalArgumentException("Invalid username."));
}
这样写参数合法性检测,应该足够优雅了吧。