概述
最近项目组内做code review,充斥着大量的、原始的、丑陋的判空语句,大致类似下面的代码:
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();
}
}
}
}
让整体的代码显得十分的臃肿庞大丑陋,那么怎么办呢?其实一行代码就可以搞定:
String result = Optional.ofNullable(user)
.map(u -> u.getAddress())
.map(a -> a.getCountry())
.map(c -> c.getIsocode())
.orElse("default");
利用Optional这个jdk8中引入的类就可以优雅的处理,现在我们来详细讲解下这个类的使用和源码。
创建Optional实例
Optional类, 是对value值进行了包装,它的值可能是null, 也可能不是null
一共有两个方法创建Optional实例
static Optional of(T value)
- 说明:
传入的value为null,会抛错,也就是说value不许不为null - 演示:
@Test
public void testOf() {
Integer value = 2;
// 正常
Optional<Integer> op = Optional.of(value);
value = null;
// 报空指针
op = Optional.of(value);
}
- static Optional ofNullable(T value)
- 说明:
传入的value可能是null, 也可能不是null - 演示:
@Test
public void testOfNullable() {
Integer value = 2;
// 正常
Optional<Integer> op = Optional.ofNullable(value);
value = null;
// 不报错
op = Optional.ofNullable(value);
}
获取Optional中的值
以前创建了Optional实例,现在你可能要获取里面的value, 有下面几个方法。
- T get()
- 说明:
最朴素的获取原生value的方法,如果value是空,则直接抛出异常,否则返回。 - 演示:
@Test
public void testGet() {
Integer value = 2;
// 正常
Optional<Integer> op = Optional.ofNullable(value);
Integer opVal = op.get();
Assert.assertEquals(opVal, value);
op = Optional.ofNullable(null);
// 会抛出异常
op.get();
}
- 源码:
- T orElse(T other)
- 说明:
如果值存在返回,否则返回orElse中传入的other - 演示:
@Test
public void testOrElse() {
// 正常
Optional<Integer> op = Optional.ofNullable(2);
Integer opVal = op.orElse(3);
Assert.assertEquals(opVal, new Integer(2));
op = Optional.ofNullable(null);
// 为空,则返回3
opVal = op.orElse(3);
Assert.assertEquals(opVal, new Integer(3));
}
- 源码:
- T orElseGet(Supplier<? extends T> other)
- 说明:
如果存在则返回该值,否则调用other这个函数编程并返回该调用的结果。 - 演示:
@Test
public void testOrElseGet() {
// 正常
Optional<Integer> op = Optional.ofNullable(2);
Integer opVal = op.orElseGet(()-> {
return new Integer(3);
});
Assert.assertEquals(opVal, new Integer(2));
op = Optional.ofNullable(null);
// 为空,则返回3
opVal = op.orElseGet(()-> {
return new Integer(3);
});
Assert.assertEquals(opVal, new Integer(3));
}
-
问题: orElseGet和orElse有什么区别吗?
orElse() 和 orElseGet() 的不同之处在于当 ofNullable() 传入参数不为空时,orElse() 方法仍然创建了 other这个 对象。与之相反,orElseGet() 方法不创建对象。在执行较密集的调用时,比如调用 Web 服务或数据查询,这个差异会对性能产生重大影响。
而且我们还可以在orElseGet方法中加些日志,可以把这种为空的异常情况暴露出来。
- T orElseThrow(Supplier<? extends X> exceptionSupplier)
- 说明:
如果存在则返回该值,否则为空的话可以抛出自定义的异常。 - 演示:
@Test
public void testOrElseThrow() {
Optional<Integer> op = Optional.ofNullable(null);
// 为空,则抛出指定的异常类型
Integer opVal = op.orElseThrow(()-> {
return new RuntimeException();
});
// 或抛出runtime异常
Assert.assertEquals(opVal, new Integer(3));
}
判断Optional是否为空
-
boolean isPresent()
-
说明:
判断value是否为空 -
演示:
@Test
public void testIsPresent() {
Optional<Integer> op = Optional.ofNullable(null);
// 为空
Assert.assertFalse(op.isPresent());
}
- void ifPresent(Consumer<? super T> consumer)
- 说明:
如果存在值,则调用对应的consumer方法,否则不执行任何操作。 - 演示:
@Test
public void testIfPresent() {
Optional<Integer> op = Optional.ofNullable(5);
op.ifPresent(value -> {
// 如果存在,打印出来
System.out.println(value);
});
}
Optional中的过滤、转换方法
此外Optional额外还提供了一些过滤、转换的方法。
- Optional filter(Predicate<? super T> predicate)
- 说明:
对Optional中的value进行过滤,如果不匹配,返回空 - 演示:
@Test
public void testFilter() {
// 不满足过滤条件,返回空 Optional
Optional<String> op = Optional.ofNullable("10").filter(item -> "15".equals(item));
Assert.assertFalse(op.isPresent());
}
- Optional map(Function<? super T, ? extends U> mapper)
小结: 回到概述的案例,就是通过不断的map, 链式调用返回内层对象的值。
- Optional flatMap(Function<? super T, Optional> mapper)
小结: 如果对于返回值非Optional类型,可以用map方法, 否则使用flatMap更加方便
总结
本文讲解了Optional这个类的使用和源码,非常的简单,实用,大家可以在自己的项目中用起来了。是代码更加优雅!有问题大家可以在评论区留言~
参考
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/132110.html