java 8中的Optional

原创性声明:本文完全为笔者原创,请尊重笔者劳动力。转载务必注明原文地址。

2014年java8发布后,许多新特性被广泛应用。个人认为,最高频出现的就是以下三个新东西:

1.Optional
2.新的日期类Calendar
3.Lambda表达式

今天我先整理一下Optional类。它有以下几个APi方法:

1. empty
2. equal
3. filter
4. flatMap
5. get
6. hashCode
7. ifPresent
8. map
9. of
10. ofNullable
11. orElse
12. orElseGet
13. orElseThrow
14. toString
这个类的作用是什么呢?(下面针对几个常用的举例说明)

解决java开发人员避免空指针异常而做的繁琐的if...else处理。

比如,有一个Article类,它具有属性name,我们要判断name属性是否等于"java 8中的Optional",我们可以这样做:

Article article = articleRepository.findOne(id);
if (article != null) {
  if (article.getName() != null) {
    if (article.getName().equal("java 8中的Optional")) {
       return true;
    } else {
       return false;
    }
  } else {
    return false;
  }
} else {
  return false;
}

这没毛病,而且可以说代码非常严谨。只是很多if...else看上去并不好受,而且其实上面的代码看上去有很多,而功能却很简单,假设上面不存在空指针问题,代码可以精简如下:

Article article = articleRepository.findOne(id);
if (article.getName().equal("java 8中的Optional")) {
    return true;
} else {
    return false;
}

但事实我们不得不去判断空指针异常,因为数据库中可能不存在id对应的article,即使存在,它的name可能也是null。但是Optional类允许我们这样做:

return Optional.ofNullable(articleRepository.findOne(id)).map(c -> c.getName())
    .map(name -> name.equal("java 8中的Optional"))
    .orElse(false);

三行代码,精简、严谨、优雅。

虽然可能初次接触Optional可能会有点不容易理解。

上面的结构其实可以这样去理解:

return Optional.ofNullable(...) //1
      .map(...) //2
      .map(...) //3
      .orElse(...); //4

下面详细解释一下:

括号1中代表初始数据(article),作为参数传入OfNullable方法,这个方法其实是构建一个Optional封装类,把数据articleRepository.findOne(id)的返回值包装进去,如果这个返回值是null,直接执行orElse方法,也就是取false,相当于直接return false,如果不是null,执行第一个map方法,map方法的参数是一个Lambda表达式,c代表了Optional包装的非空数据(即articleRepository.findOne(id)的返回值,已经确保了非空),接着c.getName()将会被执行,返回值将会继续被封装成Optional,如果这个返回值(name)为null,将会执行orElse,结果仍为return false,如果非空,继续下一个map方法,同理,第二个map方法也是一个Lambda表达式, 第一个map方法的返回值(Optional)所包装的非空数据(name)将会被传入第二个map方法的name参数中,然后继续执行name.equal("java 8中的Optional"),后面没有map方法了,所以这个语句的结果将会被作为整个Optional操作的结果。

上面的解释就是这样一个Optional操作的详细解释,所以显然,map方法可以无限多。虽然解释看上去很复杂,但是习惯了这样的思维后,其实也就那回事儿。

另外,map方法里是一个Lambda表达式,所以,它里面可以做更多的事情,比如,我们希望再获取article对象后,更改一下author属性,再持久化操作, 可以这样去修改第一个map方法:

return Optional.ofNullable(articleRepository.findOne(id))
    .map(c -> {
          c.setAuthor("dongfangyihaolan");
          articleRepository.saveAndFlush(c);
          return c.getName()
    })
    .map(name -> name.equal("java 8中的Optional"))
    .orElse(false);

它并不影响整个name的判断,也很好的处理了更新article对象时的空指针问题。

tip: 注意,当Lambda表达式是语句块(有{}时),里面需要显式的加上return , return的数据会被封装成Optional,其包装的数据会被作为参数传递至下一个map, 并执行下一轮map

常用的方法还有一个filter,过滤。仍以上面为例,我们希望只对有效的article执行后面的map操作,该如何?

你当然可以在jpa方法上做过滤查询或者在return c.getName()前做是否有效判断,但是现在用filter也可以做到。如下:

return Optional.ofNullable(articleRepository.findOne(id))
    .filter(c -> ifEffect(c))
    .map(c -> c.getName())
    .map(name -> name.equal("java 8中的Optional"))
    .orElse(false);

ifEffect()是自定义判断是否有效的方法。如果为false,也将执行orElse,为true才去继续执行map

同样,filter也可以用语句块的形式,完成更多的处理。

还有ifPresentget也值得一提,例如一个article被封装好成为了Optional,假设为articleOpt,可以用articleOpt.ifPresent判断是否为null,用articleOpt.get将封装的数据实体取出。

更多关于Optional的内容还是自己看看官方API吧。

你可能感兴趣的:(java 8中的Optional)