Convert a List into an Option if it is populated

2020-02-09 07:45发布

问题:

I have a method that should convert a list to an Option of an object, or None if the list is empty.

def listToOption(myList: List[Foo]): Option[Bar] = {
  if(myList.nonEmpty) Some(Bar(myList))
  else None
}

case class Bar(fooList: List[Foo]) {}

For some reason, my solution feels rather inelegant, and not the Scala way. It seems I should be able to use a method on List to do this sort of thing, but I can't wrap my head around it.

Is there a more Scala-like way to do this?

回答1:

Lee's answer is good, but I think this corresponds to the intention a bit more clearly:

Option(myList).filter(_.nonEmpty).map(Bar)


回答2:

myList.headOption.map(_ => Bar(myList))


回答3:

Starting Scala 2.13, Option has a when builder:

Option.when(condition)(result)

which in our case gives:

Option.when(myList.nonEmpty)(Bar(myList))
// val myList = List[Int]()    =>    Option[Bar] = None
// val myList = List(1, 2)     =>    Option[Bar] = Some(Bar(List(1, 2)))

Also note Option.unless which promotes the opposite condition:

Option.unless(myList.isEmpty)(Bar(myList))
// val myList = List[Int]()    =>    Option[Bar] = None
// val myList = List(1, 2)     =>    Option[Bar] = Some(Bar(List(1, 2)))


回答4:

import scalaz._; import Scalaz._
myList.toNel.map(Bar)

toNel - is "to non-empty list" here, it returns Option[NonEmptyList] for safety:

scala> case class Bar(a: NonEmptyList[Int])
defined class Bar

scala> List(1,2,3).toNel.map(Bar)
res64: Option[Bar] = Some(Bar(NonEmptyList(1, 2, 3)))

scala> List[Int]().toNel.map(Bar)
res65: Option[Bar] = None


回答5:

How about:

Some(myList) collect { case(l@hd::tl) => Bar(l) }

Seems pretty scala-esque to me.



标签: scala