偏函数是仅对您可能传递给它的那些类型的值的子集有效的函数。例如:
val root: PartialFunction[Double,Double] = {
case d if (d >= 0) => math.sqrt(d)
}
scala> root.isDefinedAt(-1)
res0: Boolean = false
scala> root(3)
res1: Double = 1.7320508075688772
当您知道如何检查函数是否已定义时,这很有用。收集,例如:
scala> List(0.5, -0.2, 4).collect(root) // List of _only roots which are defined_
res2: List[Double] = List(0.7071067811865476, 2.0)
这不会帮助您将两个参数放在您真正想要的位置。
相反,部分应用的函数是其中一些参数已经被填充的函数。
def add(i: Int, j: Int) = i + j
val add5 = add(_: Int,5)
现在你只需要一个参数——添加 5 的东西——而不是两个:
scala> add5(2)
res3: Int = 7
你可以从这个例子中看到如何使用它。
但是,如果您需要指定这两个参数,这仍然不会这样做 - 例如map
,您想使用 ,并且您需要给它一个参数的函数,但您希望它添加两个不同的东西。那么你可以
val addTupled = (add _).tupled
这将部分应用该函数(实际上,只需从该方法中创建一个函数,因为没有填充任何内容),然后将单独的参数组合成一个元组。现在您可以在需要单个参数的地方使用它(假设类型正确):
scala> List((1,2), (4,5), (3,8)).map(addTupled)
res4: List[Int] = List(3, 9, 11)
相比之下,currying又不一样了。它将表单的功能(A,B) => C
转换为A => B => C
. 也就是说,给定一个具有多个参数的函数,它将生成一个函数链,每个函数接受一个参数并返回一个较短的链(您可以将其视为一次部分应用一个参数)。
val addCurried = (add _).curried
scala> List(1,4,3).map(addCurried)
res5: List[Int => Int] = List(<function1>, <function1>, <function1>)
scala> res5.head(2) // is the first function, should add 1
res6: Int = 3
scala> res5.tail.head(5) // Second function should add 4
res7: Int = 9
scala> res5.last(8) // Third function should add 3
res8: Int = 11