和重载一样currying吗?

时间:2012-05-16 00:42:46

标签: oop functional-programming overloading currying

函数式编程与OO编程的重载相同吗?如果没有,为什么? (如有可能,请附上例子)

韩国社交协会

3 个答案:

答案 0 :(得分:12)

不,它们完全不相关且不相似。

Overloading是一种允许在不同类型中使用相同代码的技术 - 通常在函数式编程中称为多态(各种形式)。

多态函数:

 map :: (a -> b) -> [a] -> [b]
 map f []     = []
 map f (x:xs) = f x : map f xs

这里,map是一个可以在任何列表上运行的函数。它是多态的 - 它与Int列表一样可以作为哈希表树的列表。它也是高阶,因为它是一个以函数作为参数的函数。

Currying是将一个 n 参数结构的函数转换为一个函数链,每个函数都取一个参数。

在curried语言中,你可以将任何函数应用于它的一些参数,产生一个接受其余参数的函数。部分应用的函数是闭包。

你可以通过应用CurrySchonfinkel发明的转换,将一个curried函数转换为一个未经证实的函数(反之亦然)。

curry :: ((a, b) -> c) -> a -> b -> c 
   -- curry converts an uncurried function to a curried function.

uncurry :: (a -> b -> c) -> (a, b) -> c
   -- uncurry converts a curried function to a function on pairs.

答案 1 :(得分:11)

Currying并非特定于函数式编程,而且重载并非特定于面向对象的编程。

"柯里"是使用函数,您可以传递少于获取其余参数的函数所需的参数。即如果我们有一个函数plus,它接受​​两个整数参数并返回它们的总和,那么我们可以将单个参数1传递给plus,结果是为事物添加1的功能。

在Haskellish语法中(通过邻接函数应用):

plusOne = plusCurried 1
three = plusOne 2
four = plusCurried 2 2
five = plusUncurried 2 3

模糊的Cish语法(用括号括起功能):

plusOne = plusCurried(1)
three = plusOne(2)
four = plusCurried(2)(2)
five = plusUncurried(2, 3)

您可以在这两个示例中看到仅在1个参数上调用plusCurried,结果是可以绑定到变量然后在另一个参数上调用的结果。您之所以考虑将currying作为函数式编程概念,是因为它在函数式语言中看到了最多的用法,其语法通过邻接来应用,因为语法中的currying变得非常自然。 plusCurriedplusUncurried在Haskellish语法中定义fourfive的应用程序合并为完全无法区分,因此您可以将所有函数始终完全用于curry(即让每个函数都只是一个参数的函数,只有其中一些函数会返回其他函数,然后可以应用于更多的参数)。在带有括号参数列表的应用程序的Cish语法中,fourfive的定义看起来完全不同,因此您需要区分plusCurriedplusUncurried。此外,导致当今面向对象语言的命令式语言从未能够将函数绑定到变量或将它们传递给其他函数(这被称为具有一流的函数),并且没有那种设施和# 39;除了在所有参数上调用它之外,你实际上没有任何东西可以除了curried函数,所以没有任何意义。今天的一些OO语言仍然没有一流的功能,或者最近才获得它们。

术语currying还指将多个参数的函数转换为一个参数并获取单个参数并返回另一个函数(它可以返回另一个函数......)和&#的过程。 34; uncurrying"可以参考进行逆向转换的过程。


重载是一个完全不相关的概念。重载名称意味着给出具有不同特征的多个定义(参数类型,参数数量,返回类型等),并让编译器通过其出现的上下文解析名称的给定外观所指的定义。

一个相当明显的例子是我们可以定义plus来添加整数,但也使用相同的名称plus来添加浮点数,我们可以使用它来连接字符串,数组,列表等,或添加向量或矩阵。就语言实现而言,所有这些都具有非常不同的实现,彼此无关,但我们碰巧给它们提供了相同的名称。然后编译器负责确定plus stringA stringB应该调用字符串plus(并返回一个字符串),而plus intX intY应该调用整数plus(并返回一个整数) )。

同样,没有内在的理由说明这个概念是一个" OO概念"而不是功能编程概念。它恰好发生在非常自然地适用于开发的静态类型的面向对象语言中;如果您已经解析了调用该方法的对象调用哪个方法,那么它只是一个小范围以允许更一般的重载。完全临时的重载(你只做多次定义相同的名称并信任编译器来解决它)并不适合用于具有一等函数的语言,因为当你传递重载的名字时作为一个函数本身,你没有调用上下文来帮助你找出预期的定义(如果他们真正想要的是通过所有重载的定义,程序员可能会感到困惑)。 Haskell开发了类型类作为使用重载的更有原则的方式;这些 do 允许您一次传递所有重载的定义,并允许类型系统表达类似于"函数f和{的任何类型{1}}已定义"。


总结:

  • currying and overloading完全不相关
  • currying是关于将函数应用于少于它们所需的参数以获得剩余参数的函数
  • 重载是为同一个名称提供多个定义,让编译器选择每次使用名称时使用哪个定义
  • 既不是currying也不是overloading都不是特定于函数式编程或面向对象的编程;由于语言的发展方式使得它们在一种语言中更有用或更明显,它们在各种历史语言中恰好更为普遍。

答案 2 :(得分:2)

重载具有多个具有相同名称的函数,具有不同的参数。

Currying是您可以获取多个参数的地方,并有选择地设置一些参数,例如,您可能只有一个变量。

因此,如果你有三维的图形功能,你可能有: justgraphit(double[] x, double[] y, double[] z),您想要绘制图表。

通过currying你可以: var fx = justgraphit(xlist)(y)(z)您现在已经设置了fx,现在它有两个变量。

然后,稍后,用户选择另一个轴(日期)并设置y,所以现在你有: var fy = fx(ylist)(z)

然后,稍后通过循环一些数据来绘制信息的图形,唯一的变化是z参数。

这使得复杂的函数变得更简单,因为您不必继续传递大部分设置的变量,因此可读性会增加。