2017-03-01 114 views
5

我有两个选择组合两个选项[列表[字符串]在斯卡拉

val opt1 = Some(List("Sal", "Salil")) 
val opt2 = Some(List("Sal2", "Salil2")) 

无论OPT1或OPT2可能是无。如果其中一个是None,那么我需要一个包含在其他列表中的选项。如果两者都是None,则应返回None。

如果两者都有些则用一些含两份名单的元素的列表,如下图所示:

Some(List(Sal, Salil, Sal2, Salil2)) 

我知道我可以自己动手完成这一点,但有一个优雅的方式来做到这一点?如果其中一个选项为无,则理解不起作用。

回答

6
Option((opt1 ++ opt2).flatten.toList).filter(_.nonEmpty) 
+2

这不会返回正确的值,如果两个输入都为'一些(名单())'。这应该返回'Some(List())',但是你的解决方案返回'None'。 –

5

你可以做到这一点很好地使用半群与scalaz或猫附加:

import scalaz._, Scalaz._ // for cats use `import cats._, implicits._` 

val opt1 = Option(List("Sal", "Salil")) 
val opt2 = Option(List("Sal2", "Salil2")) 

scala> opt1 |+| opt2 
res0: Option[List[String]] = Some(List(Sal, Salil, Sal2, Salil2)) 

scala> opt1 |+| None 
res1: Option[List[String]] = Some(List(Sal, Salil)) 

scala> Option.empty[List[String]] |+| None 
res2: Option[List[String]] = None 

否则,标准库,你可能需要处理其案件逐案:

(opt1, opt2) match { 
    case (Some(a), Some(b)) => Option(a ++ b) 
    case (Some(a), None) => Option(a) 
    case (None, Some(b)) => Option(b) 
    case _ => None 
} 

或使用收集方法将它们弄平:

scala> List(opt1, opt2).flatten.flatten 
res5: List[String] = List(Sal, Salil, Sal2, Salil2) 

scala> List(opt1, None).flatten.flatten 
res6: List[String] = List(Sal, Salil) 
3

我不认为只有一个,适当的,优雅的方式来实现这一点。 还有就是我的命题:

val opt1 = Some(List("Sal", "Salil")) 
val opt2 = Some(List("Sal2", "Salil2")) 

def merge(xs: Option[Iterable[_]]*) = xs.flatten.reduceLeftOption(_ ++ _) 

的结果:

merge (opt1, opt2) 
res1: Option[Iterable[_]] = Some(List(Sal, Salil, Sal2, Salil2)) 

merge (None, opt2) 
res2: Option[Iterable[_]] = Some(List(Sal2, Salil2)) 

merge (opt1, None) 
res5: Option[Iterable[_]] = Some(List(Sal, Salil)) 

merge (None, None) 
res6: Option[Iterable[_]] = None 
相关问题