24

がある限り、PartialFunction[X,R]それを返す関数に変換するのは非常に簡単ですOption[R]

def pfToOptf[X, R](f: PartialFunction[X,R])(x: X) =
    if (f.isDefinedAt(x)) Some(f(x))
    else None

ただし、タスクが逆の場合はどうなるでしょうか。引数としてf取得し、結果として返す関数があるとします。そして、私はそれを作りたいと思っています。最善の方法は何ですか?XOption[R]PartialFunction[X,R]

私が思いついたものは、私の好みにはかなり醜く見えます:

def optfToPf[X,R](f: X => Option[R]) : PartialFunction[X,R] = {
    object extractor {
        def unapply(x: X): Option[R] = f(x)
    }

    { case extractor(r) => r }
}

私が逃したより良い方法はありますか?

4

3 に答える 3

35

を開始Scala 2.9すると、Function.unlift正確に次のようになります。

def unlift[T, R](f: (T) => Option[R]): PartialFunction[T, R]

関数 T => Option[R] を PartialFunction[T, R] に変換します。

于 2013-12-20T07:04:01.363 に答える
12

これはどう:

Welcome to Scala version 2.8.0.r19650-b20091114020153 (Java HotSpot(TM) Client VM, Java 1.6.0_17).
Type in expressions to have them evaluated.
Type :help for more information.

scala> def optfToPf[X,R](f: X => Option[R]): PartialFunction[X,R] = x => f(x) match {
     |     case Some(r) => r
     | }
optfToPf: [X,R](f: (X) => Option[R])PartialFunction[X,R]

scala>
于 2009-12-15T16:01:45.127 に答える
5

apply と isDefinedAt を手動でオーバーライドできると思いますが、見苦しい方法で行います。

def optfToPf[X,R](f: X => Option[R]) = new PartialFunction[X,R] {
  def apply(x: X): R = f(x).get
  def isDefinedAt(x: X): Boolean = f(x) != None
}

テスト:

scala> val map = Map(1 -> 2)
map: scala.collection.immutable.Map[Int,Int] = Map(1 -> 2)

scala> map(1)
res0: Int = 2

scala> def mapOpt(key: Int) = map.get(key)
mapOpt: (key: Int)Option[Int]

scala> mapOpt(1)
res1: Option[Int] = Some(2)

scala> mapOpt(2)
res2: Option[Int] = None

scala> val mapPf = optfToPf(mapOpt _)
mapPf: java.lang.Object with PartialFunction[Int,Int] = <function1>

scala> mapPf.isDefinedAt(2)
res3: Boolean = false

scala> mapPf.isDefinedAt(1)
res4: Boolean = true

scala> mapPf(1)
res5: Int = 2

scala> mapPf(2)
java.util.NoSuchElementException: None.get
于 2009-12-15T16:53:04.303 に答える