Simplify if (x) Some(y) else None?

2020-05-25 06:50发布

问题:

This common pattern feels a bit verbose:

if (condition) 
  Some(result)
else None

I was thinking of using a function to simplify:

def on[A](cond: Boolean)(f: => A) = if (cond) Some(f) else None

This reduces the top example to:

on (condition) { result }

Does something like this exist already? Or is this overkill?

回答1:

Scalaz includes the option function:

import scalaz.syntax.std.boolean._

true.option("foo") // Some("foo")
false.option("bar") // None


回答2:

You could create the Option first and filter on that with your condition:

Option(result).filter(condition)

or if condition is not related to result

Option(result).filter(_ => condition)


回答3:

Starting Scala 2.13, Option is now provided with the when builder which does just that:

Option.when(condition)(result)

For instance:

Option.when(true)(45)
// Option[Int] = Some(45)
Option.when(false)(45)
// Option[Int] = None

Also note the coupled unless method which does the opposite.



回答4:

You can use the PartialFunction companion object and condOpt:

PartialFunction.condOpt(condition) {case true => result}

Usage:

 scala> PartialFunction.condOpt(false) {case true => 42}
 res0: Option[Int] = None

 scala> PartialFunction.condOpt(true) {case true => 42}
 res1: Option[Int] = Some(42)


回答5:

import scalaz._, Scalaz._
val r = (1 == 2) ? Some(f) | None
System.out.println("Res = " + r)


回答6:

Here is another approach that is quite straightforward:

Option(condition).collect{ case true => result }

A simple example:

scala> val enable = true
enable: Boolean = true

scala> Option(enable).collect{case true => "Yeah"}
res0: Option[String] = Some(Yeah)

scala> Option(!enable).collect{case true => "Yeah"}
res1: Option[String] = None

Here some advanced non-Boolean examples that put the condition into the pattern match:

val param = "beta"
Option(param).collect{case "alpha" => "first"} // gives None

Option(param).collect{case "alpha" => "first"
                      case "beta"  => "second"
                      case "gamma" => "third"} // gives Some(second)

val number = 999
Option(number).collect{case 0 => "zero"
                       case x if x > 10 => "too high"} // gives Some(too high)


回答7:

Similar to Scalaz, the Typelevel cats ecosystem has the mouse package with option:

scala> true.option("Its true!")
res0: Option[String] = Some(Its true!)


标签: scala