2017-03-01 7 views
5

に私は2つのオプションは2つのオプション[一覧[文字列]] Scalaの

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

OPT1またはOPT2のどちらかがNoneに可能性を持っているを組み合わせます。いずれかがNoneの場合、Listを含むオプションを他のListに含める必要があります。両方がNoneの場合、Noneを返す必要があります。

両方が、その後いくつか以下に示すように両方のリストから要素を含むリストを持ついくつかある場合:

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

私は手でこれを行うことができます知っているが、これを行うためのエレガントな方法は何ですか?オプションの1つが「なし」の場合、補題は機能しません。

答えて

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 
関連する問題