JDK8新特性:使用Optional避免null导致的NullPointerException

空指针异常是导致Java应用程序失败的最常见原因。以前,为了解决空指针异常,Google公司著名的Guava项目引入了Optional类,Guava通过使用检查空值的方式来防止代码污染,它鼓励程序员写更干净的代码。受到Google Guava的启发,Optional类已经成为Java 8类库的一部分。Optional实际上是个容器:它可以保存类型T的值,或者仅仅保存null。Optional提供很多有用的方法,这样我们就不用显式进行空值检测。


Optional.of()或者Optional.ofNullable():创建Optional对象,差别在于of不允许参数是null,而ofNullable则无限制。

  1. // 参数不能是null
  2. Optional<Integer> optional1 = Optional.of( 1);
  3. // 参数可以是null
  4. Optional<Integer> optional2 = Optional.ofNullable( null);
  5. // 参数可以是非null
  6. Optional<Integer> optional3 = Optional.ofNullable( 2);



Optional.empty():所有null包装成的Optional对象:
  1. Optional<Integer> optional1 = Optional.ofNullable( null);
  2. Optional<Integer> optional2 = Optional.ofNullable( null);
  3. System.out.println(optional1 == optional2); // true
  4. System.out.println(optional1 == Optional.<Integer>empty()); // true
  5. Object o1 = Optional.<Integer>empty();
  6. Object o2 = Optional.<String>empty();
  7. System.out.println(o1 == o2); // true



isPresent():判断值是否存在
  1. Optional<Integer> optional1 = Optional.ofNullable( 1);
  2. Optional<Integer> optional2 = Optional.ofNullable( null);
  3. // isPresent判断值是否存在
  4. System.out.println(optional1.isPresent() == true);
  5. System.out.println(optional2.isPresent() == false);


ifPresent(Consumer consumer):如果option对象保存的值不是null,则调用consumer对象,否则不调用
  1. Optional<Integer> optional1 = Optional.ofNullable( 1);
  2. Optional<Integer> optional2 = Optional.ofNullable( null);
  3. // 如果不是null,调用Consumer
  4. optional1.ifPresent( new Consumer<Integer>() {
  5. @Override
  6. public void accept(Integer t) {
  7. System.out.println( "value is " + t);
  8. }
  9. });
  10. // null,不调用Consumer
  11. optional2.ifPresent( new Consumer<Integer>() {
  12. @Override
  13. public void accept(Integer t) {
  14. System.out.println( "value is " + t);
  15. }
  16. });



orElse(value):如果optional对象保存的值不是null,则返回原来的值,否则返回value
  1. Optional<Integer> optional1 = Optional.ofNullable( 1);
  2. Optional<Integer> optional2 = Optional.ofNullable( null);
  3. // orElse
  4. System.out.println(optional1.orElse( 1000) == 1); // true
  5. System.out.println(optional2.orElse( 1000) == 1000); // true



orElseGet(Supplier supplier):功能与orElse一样,只不过orElseGet参数是一个对象
  1. Optional<Integer> optional1 = Optional.ofNullable( 1);
  2. Optional<Integer> optional2 = Optional.ofNullable( null);
  3. System.out.println(optional1.orElseGet(() -> {
  4. return 1000;
  5. }) == 1); //true
  6. System.out.println(optional2.orElseGet(() -> {
  7. return 1000;
  8. }) == 1000); //true


orElseThrow():值不存在则抛出异常,存在则什么不做,有点类似Guava的Precoditions
  1. Optional<Integer> optional1 = Optional.ofNullable( 1);
  2. Optional<Integer> optional2 = Optional.ofNullable( null);
  3. optional1.orElseThrow(()->{ throw new IllegalStateException();});
  4. try
  5. {
  6. // 抛出异常
  7. optional2.orElseThrow(()->{ throw new IllegalStateException();});
  8. }
  9. catch(IllegalStateException e )
  10. {
  11. e.printStackTrace();
  12. }


filter(Predicate):判断Optional对象中保存的值是否满足Predicate,并返回新的Optional。
  1. Optional<Integer> optional1 = Optional.ofNullable( 1);
  2. Optional<Integer> optional2 = Optional.ofNullable( null);
  3. Optional<Integer> filter1 = optional1.filter((a) -> a == null);
  4. Optional<Integer> filter2 = optional1.filter((a) -> a == 1);
  5. Optional<Integer> filter3 = optional2.filter((a) -> a == null);
  6. System.out.println(filter1.isPresent()); // false
  7. System.out.println(filter2.isPresent()); // true
  8. System.out.println(filter2.get().intValue() == 1); // true
  9. System.out.println(filter3.isPresent()); // false



map(Function):对Optional中保存的值进行函数运算,并返回新的Optional(可以是任何类型)
  1. Optional<Integer> optional1 = Optional.ofNullable( 1);
  2. Optional<Integer> optional2 = Optional.ofNullable( null);
  3. Optional<String> str1Optional = optional1.map((a) -> "key" + a);
  4. Optional<String> str2Optional = optional2.map((a) -> "key" + a);
  5. System.out.println(str1Optional.get()); // key1
  6. System.out.println(str2Optional.isPresent()); // false


flatMap():功能与map()相似,差别请看如下代码。flatMap方法与map方法类似,区别在于mapping函数的返回值不同。map方法的mapping函数返回值可以是任何类型T,而flatMap方法的mapping函数必须是Optional。
  1. Optional<Integer> optional1 = Optional.ofNullable( 1);
  2. Optional<Optional<String>> str1Optional = optional1.map((a) -> {
  3. return Optional.<String>of( "key" + a);
  4. });
  5. Optional<String> str2Optional = optional1.flatMap((a) -> {
  6. return Optional.<String>of( "key" + a);
  7. });
  8. System.out.println(str1Optional.get().get()); // key1
  9. System.out.println(str2Optional.get()); // key1
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值