![Java核心技术·卷Ⅱ:高级特性(原书第10版)](https://wfqqreader-1252317822.image.myqcloud.com/cover/937/34339937/b_34339937.jpg)
1.7.4 用flatMap来构建Optional值的函数
假设你有一个可以产生Optional<T>对象的方法f,并且目标类型T具有一个可以产生Optional<U>对象的方法g。如果它们都是普通的方法,那么你可以通过调用s.f().g()来将它们组合起来。但是这种组合没法工作,因为s.f()的类型为Optional<T>,而不是T。因此,需要调用:
![](https://epubservercos.yuewen.com/F21227/18365861501241106/epubprivate/OEBPS/Images/013-2-i.jpg?sign=1739511549-ErUNfoKu8cJ5pBXdNDM0yc4Qyuzaz051-0-7397c19b80e7e8c31e4d775999944289)
如果s.f()的值存在,那么g就可以应用到它上面。否则,就会返回一个空Optional<U>。
很明显,如果有更多的可以产生Optional值的方法或Lambda表达式,那么就可以重复此过程。你可以直接将对flatMap的调用链接起来,从而构建由这些步骤构成的管道,只有所有步骤都成功时,该管道才会成功。
例如,考虑前一节中安全的inverse方法。假设我们还有一个安全的平方根:
![](https://epubservercos.yuewen.com/F21227/18365861501241106/epubprivate/OEBPS/Images/013-3-i.jpg?sign=1739511549-xHtlf1Al1WaFI3feE8s8WOVbkIINUpc2-0-ac17be07894116cba7608739021e2817)
那么你可以像下面这样计算倒数的平方根了:
![](https://epubservercos.yuewen.com/F21227/18365861501241106/epubprivate/OEBPS/Images/014-i.jpg?sign=1739511549-UcPydaJXtuOLgv0Ax0NjhmIBOJOHCIaU-0-5f2edfbc72bb7bcbf4ab024b96ee1127)
或者,你可以选择下面的方式:
![](https://epubservercos.yuewen.com/F21227/18365861501241106/epubprivate/OEBPS/Images/014-2-i.jpg?sign=1739511549-GKkBmnGj2SWcj1bgkR5CsydiM5xkxP0l-0-25be257f1167d146403125237bdddcc6)
无论是inverse方法还是squareRoot方法返回Optional.empty(),整个结果都会为空。
注意:你已经在Stream接口中看到过flatMap方法(参见1.3节),当时这个方法被用来将可以产生流的两个方法组合起来,其实现方式是摊平由流构成的流。如果将可选值当作尺寸为0和1的流来解释,那么Optional.flatMap方法与其操作方式一样。
程序清单1-3中的示例程序演示了Optional API的使用方式。
程序清单1-3 optional/OptionalTest.java
![](https://epubservercos.yuewen.com/F21227/18365861501241106/epubprivate/OEBPS/Images/014-3-i.jpg?sign=1739511549-4Wbtb3hyAoGjIcRhw2sE1yZKthcE9Or9-0-7108d796588b007cf90ba4efe8cf9d9b)
![](https://epubservercos.yuewen.com/F21227/18365861501241106/epubprivate/OEBPS/Images/015-i.jpg?sign=1739511549-wELsmEOzX6zSYDI165rSkyMb83HFX9Tu-0-c4305421eff6ec6f4e0e77c439f55908)
java.util.Optional 8
·<U>Optional<U>flatMap(Function<?super T,Optional<U>>mapper)
产生将mapper应用于当前的Optional值所产生的结果,或者在当前Optional为空时,返回一个空Optional。