2014-10-26 3 views
15

Ich möchte eine List[Either[A, B]] in zwei Listen aufteilen.So teilen Sie eine Liste [Entweder [A, B]]

Gibt es einen besseren Weg?

def lefts[A, B](eithers : List[Either[A, B]]) : List[A] = eithers.collect { case Left(l) => l} 
def rights[A, B](eithers : List[Either[A, B]]) : List[B] = eithers.collect { case Right(r) => r} 

Antwort

21

nicht sicher, dass dies wirklich viel ordentlicher, aber:

scala> def splitEitherList[A,B](el: List[Either[A,B]]) = { 
     val (lefts, rights) = el.partition(_.isLeft) 
     (lefts.map(_.left.get), rights.map(_.right.get)) 
     } 
splitEitherList: [A, B](el: List[Either[A,B]])(List[A], List[B]) 

scala> val el : List[Either[Int, String]] = List(Left(1), Right("Success"), Left(42)) 
el: List[Either[Int,String]] = List(Left(1), Right(Success), Left(42)) 

scala> val (leftValues, rightValues) = splitEitherList(el) 
leftValues: List[Int] = List(1, 42) 
rightValues: List[String] = List("Success") 
+2

Edit:. (.. Lefts.map (_ left.get), rights.map (_ right.get)) wird (lefts.flatMap (_ left.toOption), rights.flatMap (_. right.toOption)) – Lombric

7

Sie tun können, es mit:

val (lefts, rights) = eithers.foldRight((List[Int](), List[String]()))((e, p) => e.fold(l => (l :: p._1, p._2), r => (p._1, r :: p._2))) 
1

Nun, in dem Fall hat es keine Eins-sein -Liner ... dann kann es ein Kinderspiel sein.

def split[A,B](eithers : List[Either[A, B]]):(List[A],List[B]) = { 
    val lefts = scala.collection.mutable.ListBuffer[A]() 
    val rights = scala.collection.mutable.ListBuffer[B]() 
    eithers.map { 
    case Left(l) => lefts += l 
    case Right(r) => rights += r 
    } 
    (lefts.toList, rights.toList) 
} 

Aber, um ehrlich zu sein, würde ich Marth ‚s Antwort bevorzugen :)

7

Wenn scalaz eine Ihrer Abhängigkeiten ich einfach separate verwenden würde:

import scalaz.std.list._ 
import scalaz.std.either._ 
import scalaz.syntax.monadPlus._ 

val el : List[Either[Int, String]] = List(Left(1), Right("Success"), Left(42)) 

scala> val (lefts, rights) = el.separate 
lefts: List[Int] = List(1, 42) 
rights: List[String] = List(Success) 
+3

Möchten Sie eine umfassendere Antwort mit allen erforderlichen Importen geben? Und die spezielle scalaz Version? – noahlz

+1

'libraryDependencies + =" org.scalaz "%%" scalaz-core "%" 7.2.2 "' und 'import scalaz.Scalaz._'. Weitere Informationen: http://stackoverflow.com/questions/36878459/how-to-turn-a-list-of-eithers-to-a-eine-von-lists-using-scalaz-monadplus-separa/36883766#36883766 –

+1

Und wenn Sie 'Katzen' verwenden, haben Sie auch' getrennt'! http://typelevel.org/cats/api/cats/syntax/SeparateOps.html#separate(implicitF:cats.MonadCombine[F],implicitG:cats.Bifoldable[G]):(F[A],F[B ]) –

5

A kompakte, aber keine CPU-wirksame Lösung:

val lefts = list.flatMap(_.left.toOption) 
val rights = list.flatMap(_.right.toOption) 
0

Eine etwas funktionelle Lösung für Seq.

def partition[A, B](seq: Seq[Either[A, B]]): (Seq[A], Seq[B]) = { 
    seq.foldLeft[(Seq[A], Seq[B])]((Nil, Nil)) { case ((ls, rs), next) => 
    next match { 
     case Left(l) => (ls :+ l, rs) 
     case Right(r) => (ls, rs :+ r) 
    } 
    } 
}