2017-03-01 1 views
5

에 나는 아무도 없을 수 두 옵션결합이 옵션 [목록 [문자열]] 스칼라

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

하나의 OPT1 또는 OPT2 있습니다. 둘 중 하나가 None이면 List에 다른 옵션이 포함 된 옵션이 필요합니다. 둘 다 None이면 None을 반환해야합니다.

아래 그림과 같이 모두는 두 목록에서 요소를 포함하는 목록과 함께 약간 약간을 경우 :

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

는 내가 손이 작업을 수행 할 수 있습니다 알고 있지만,이 작업을 수행하는 우아한 방법이? Options 중 하나가 None 일 때 For-comprehensions은 작동하지 않습니다.

답변

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