我有以下问题。假设你有2个Optional
变量
Optional<Contact> c1 = ...
Optional<Contact> c2 = ...
以及需要2个Contact类型变量的方法
void match(Contact c1, Contact c2) {...}
你需要打开c1和c2 Optional
变量并将它们传递给match()
方法。
我的问题是“在Java 8中哪种方式最优雅?”
到目前为止,我找到了两种方法:
if (c1.isPresent() && c2.isPresent()) {
match(c1.get(), c2.get());
}
c1.ifPresent((Contact _c1) -> {
c2.ifPresent((Contact _c2) -> {
match(_c1, _c2);
});
});
在我看来,这两种方式都很糟糕。在Scala我可以这样做:
for {
contact1 <- c1
contact2 <- c2
} yield {
match(contact1, contact2);
}
有没有一种方法在Java 8中做到比我上面概述的更整洁?
您在scala中提供的解决方案只是内部使用flatMaps的语法糖。您也可以在Java 8中使用平面地图(但它没有语法糖)。
c1.flatMap(contact1 -> c2.flatMap(contact2 -> match(contact1, contact2)));
它与您提供的解决方案2几乎相同。您还可以使用来自https://github.com/aol/cyclops-react(我是贡献者之一)或任何其他功能性java 8库的applicative functor。
应用函子
Optional<String> o3 = Maybe.fromOptional(o1).ap2(String::concat).ap(o2).toOptional();
对于-理解
Do.add(o1)
.add(o2)
.yield(a->b->a.concat(b));
您可以通过以下函数将Scala for-understanding理解为Java 8中的map / flatMap:
public static <T,K,V> Optional<V> map2(Optional<T> opt1, Optional<K> opt2, BiFunction<T, K, V> f) {
Optional<V> result = opt1.flatMap(t1 -> opt2.map(t2 -> f.apply(t1, t2)));
return result;
}
然后传递你的功能匹配
如果您认为参数没有值作为例外,那么您可以像下面这样处理它们:
try {
match(c1.orElseThrow(NoVal::new), c2.orElseThrow(NoVal::new));
} catch (NoVal ex) {
...
}
如果它们不是特殊情况,那么我会选择您的第一个选项,因为它更明确地表达您的意图。在我看来它非常清晰可读,并且如果您希望在选项为空时切换到使用默认值,则很容易更改为使用orElse
。