Using Scala 2.7.7:
If I have a list of Options, I can flatten them using a for-comprehension:
val listOfOptions = List(None, Some("hi"), None)
listOfOptions: List[Option[java.lang.String]] = List(None, Some(hi), None)
scala> for (opt <- listOfOptions; string <- opt) yield string
res0: List[java.lang.String] = List(hi)
I don't like this style, and would rather use a HOF. This attempt is too verbose to be acceptable:
scala> listOfOptions.flatMap(opt => if (opt.isDefined) Some(opt.get) else None)
res1: List[java.lang.String] = List(hi)
Intuitively I would have expected the following to work, but it doesn't:
scala> List.flatten(listOfOptions)
<console>:6: error: type mismatch;
found : List[Option[java.lang.String]]
required: List[List[?]]
List.flatten(listOfOptions)
Even the following seems like it should work, but doesn't:
scala> listOfOptions.flatMap(_: Option[String])
<console>:6: error: type mismatch;
found : Option[String]
required: (Option[java.lang.String]) => Iterable[?]
listOfOptions.flatMap(_: Option[String])
^
The best I can come up with is:
scala> listOfOptions.flatMap(_.toList)
res2: List[java.lang.String] = List(hi)
... but I would much rather not have to convert the option to a list. That seems clunky.
Any advice?
To complement Arjan's answer, in Scala 2.7.7 you can use
List#flatten
, but you need to help out the type inferencer:In Scala 2.8, flatten will work:
This doesn't work in 2.7.7, however:
The collections library has been redesigned, and has improved a lot in 2.8, so perhaps you might want to try to use the latest Scala 2.8 RC and see if that makes it more easy to use for you.
If you really don't want to use the toList method, I guess you can also write it like this:
Also not a thing of beauty perhaps, but at least this works in 2.7.7.