2017-03-01 110 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 
相關問題