原创性声明:本文完全为笔者原创,请尊重笔者劳动力。转载务必注明原文地址。
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也可以用语句块的形式,完成更多的处理。
还有ifPresent
和get
也值得一提,例如一个article
被封装好成为了Optional
,假设为articleOpt
,可以用articleOpt.ifPresent
判断是否为null,用articleOpt.get
将封装的数据实体取出。
更多关于
Optional
的内容还是自己看看官方API吧。