user95947
user95947

Reputation:

Scala: How to convert tuple elements to lists

Suppose I have the following list of tuples:

val tuples = listOfStrings.map(string => {
            val split = string.split(":")
            (split(0), split(1), split(2))
        })

I would like to get the split(0) in a list, split(1) in another list and so on. A simple way this could be achieved is by writing:

list1 = tuples.map(x => x._1).toList
list2 = tuples.map(x => x._2).toList
list3 = tuples.map(x => x._3).toList

Is there a more elegant (functional) way of achieving the above without writing 3 separate statements?

Upvotes: 14

Views: 23923

Answers (5)

Régis Jean-Gilles
Régis Jean-Gilles

Reputation: 32719

This will give you your result as a list of list:

tuples.map{t => List(t._1, t._2, t._3)}.transpose

If you want to store them in local variables, just do:

val List(l1,l2,l3) = tuples.map{t => List(t._1, t._2, t._3)}.transpose

UPDATE: As pointed by Blaisorblade, the standard library actually has a built-in method for this: unzip3, which is just like unzip but for triples instead of pairs:

val (l1, l2, l3) = tuples.unzip3

Needless to say, you should favor this method over my hand-rolled solution above (but for tuples of arity > 3, this would still still apply).

Upvotes: 13

PTWithy
PTWithy

Reputation: 168

You want unzip:

scala> val (numbers, homonyms) = List(("one", "won"), ("two", "too")).unzip
numbers: List[java.lang.String] = List(one, two)
homonyms: List[java.lang.String] = List(won, too)

Upvotes: 11

Kim Stebel
Kim Stebel

Reputation: 42047

If you want something that can be used for arbitrary tuple sizes:

val tupleSize = 3
0.until(tupleSize).toList
    .map(x => (_:Product).productElement(x).asInstanceOf[String])
    .map(tuples.map(_))

Obviously, this could be expressed more elegantly if you had a List of Arrays instead.

Upvotes: 3

Kyle
Kyle

Reputation: 22258

I don't know about elegant but you could do it in one line without the intermediate step of storing the tuples. Perhaps it's a little hard to read...

(for(split <- listOfStrings.map(_.split(":")))
  yield List(split(0), split(1), split(2))).transpose

repl example:

scala> listOfStrings
res1: List[java.lang.String] = List(a:b:c, d:e:f, g:h:i)

scala> (for(split <- listOfStrings.map(_.split(":")))
     |   yield List(split(0), split(1), split(2))).transpose
res2: List[List[java.lang.String]] = List(List(a, d, g), List(b, e, h), List(c, f, i))

Upvotes: 0

BeniBela
BeniBela

Reputation: 16917

You could just write the statements in a single line.

Like

 (list1, list2, list3) = tuples.foldRight((List[String](), List[String](), List[String]()))( (a,b) => (a._1 :: b._1, a._2 :: b._2, a._3 :: b._3 ) )

Upvotes: 1

Related Questions