Swift中的咖喱功能

Enc*_*PTL 9 currying swift

我想创建一个返回咖喱函数的函数,如下所示

func addTwoNumbers(a: Int)(b: Int) -> Int {
    return a + b
}

addTwoNumbers(4)(b: 6) // Result: 10

var add4 = addTwoNumbers(4)
add4(b: 10) // returns 14     
Run Code Online (Sandbox Code Playgroud)

这种函数的返回类型是什么?如何使用采用Variadic参数的函数生成这样的函数.

func generateCurry(.../*Variadic parameters*/) -> .../*curry function type*/ {
  return ...//curry function
}
Run Code Online (Sandbox Code Playgroud)

我想要一个通用的解决方案,而不是只将Int作为参数在generateCurry函数的参数中

let curried = curry(func(a, b, c) {
  print(a + b + c)
})
curried(1)(2)(3) //prints 6
Run Code Online (Sandbox Code Playgroud)

Ada*_*arp 14

你可以通过闭包轻松实现这一目标:

/// Takes a binary function and returns a curried version
func curry<A,B,C>(f: (A, B) -> C) -> A -> B -> C {
    return { a in { b in f(a, b) } }
}

curry(+)(5)(6) // => 11

let add: Int -> Int -> Int = curry(+)
add(5)(6) // => 11
Run Code Online (Sandbox Code Playgroud)

如果能够为需要3,4个或更多参数的函数执行相同的操作,但不重复实现,那将是非常好的.这样一个函数的签名可能会开始:

/// Take a function accepting N arguments and return a curried version
func curry<T>(args: T...) -> /* ? */
Run Code Online (Sandbox Code Playgroud)

返回类型是什么?它会根据函数的输入而改变.这在目前Swift中是不可能的,如果没有某种宏系统,我认为它根本不可能.但即使使用宏,我也不认为编译器会满意,除非它在编译时知道列表的长度.

话虽如此,使用接受3,4,5或更多参数的版本手动重载currying函数是非常简单的:

func curry<A,B,C,D>(f: (A, B, C) -> D) -> A -> B -> C -> D {
    return { a in { b in { c in f(a,b,c) } } }
}

func curry<A,B,C,D,E>(f: (A, B, C, D) -> E) -> A -> B -> C -> D -> E {
    return { a in { b in { c in { d in f(a,b,c,d) } } } }
}

// etc.
Run Code Online (Sandbox Code Playgroud)