Currying and Partial application

深入理解代替单纯记忆

  • Currying和Partial application都是函数式编程范式中的概念
  • 尽管如此,在一些高级语言中(如Swift、Scala),在语法层面或者可以通过一些技术手段实现这些概念

本文将从概念和使用场景方面介绍这两个名词

概念

Currying则翻译为柯里化 ;Partial application可翻译为部分应用 看下各自的概念:

Partial application from Wiki: In computer science, partial application (or partial function application) refers to the process of fixing a number of arguments of a function, producing another function of smaller arity. Given a function f:(X×Y×Z)→N{}, we might fix (or 'bind') the first argument, producing a function of type partial(f):(Y×Z)→N{}. Evaluation of this function might be represented as fpartial(2,3){}.
Currying from Wiki: In mathematics and computer science, currying is the technique of translating a function that takes multiple arguments into a sequence of families of functions, each taking a single argument.

Partial application是指,先固定一个函数的部分参数,创建另一个函数,该函数只需接受剩余的参数

比如

swift 复制代码
// 原始函数
func multiply(_ x: Int, _ y: Int, _ z: Int) -> Int {
    return x * y * z
}

// 部分应用(固定 x)
func partialMultiply(by x: Int) -> (Int, Int) -> Int {
    return { y, z in
        return multiply(x, y, z)
    }
}

// 使用方式:
let multiplyBy2 = partialMultiply(by: 2)
let result = multiplyBy2(3, 4)  // 相当于 multiply(2, 3, 4)
print(result) // 24
  • 我们基于原函数multiply创建了partialMultiplypartialMultiply就是概念中所说的固定了x这个参数
  • 返回值仍是一个函数,类型是(Int, Int) -> Int,该函数可以接受yz参数

其实在上面Swift代码中,partialMultiply的返回值类型是Closure不是Function,但两者在这样的场景下使用是非常类似,也就是说虽然Swift中没有Partial application的语法,但可以通过Closure实现类似概念

在使用过程中能更清楚看到这一点:

  • 执行partialMultiply函数,并传入为2的x参数
  • 返回的新函数,命名为multiplyBy2
  • 后续通过执行multiplyBy2,并传入剩余的参数,就实现了让2与yz相乘

Currying指的是,将一个接受多个参数的函数,转换为一系列函数,每个函数仅接受一个参数

swift 复制代码
// 原始函数
func multiply(_ x: Int, _ y: Int, _ z: Int) -> Int {
    return x * y * z
}

func curryMultiply(_ x: Int) -> (Int) -> (Int) -> Int {
    return { y in
        return { z in
            return x * y * z
        }
    }
}

// 使用方式:
let step1 = curryMultiply(2)     // 返回 (Int) -> (Int) -> Int
let step2 = step1(3)             // 返回 (Int) -> Int
let result = step2(4)            // 返回 2 * 3 * 4 = 24

// 或者链式调用
let result2 = curryMultiply(2)(3)(4)
print(result2) // 24
  • 原函数是multiply
  • 通过curryMultiply,对原函数进行转换,返回值也是一个函数((Int) -> (Int) -> Int),其实是函数的嵌套(或者说函数的函数)
  • 在使用时能看出来,原multiply的执行,被拆分成了curryMultiplystep1step2三次执行,每次执行仅接受1个参数

能够看出来,Currying其实是Partial application的一种特殊情况

使用场景

比较明显的是,两个概念都可以用于代码的复用(或者说函数的复用),比如上面代码中的multiplyBy2函数就可以作为一个变量或者全局函数来复用

与高阶函数结合使用

swift 复制代码
func add(_ x: Int, _ y: Int) -> Int {
    return x + y
}

// Partial application:固定第一个参数
func add(_ x: Int) -> (Int) -> Int {
    return { y in x + y }
}

let add5 = add(5)
let result = [1, 2, 3].map(add5)  // [6, 7, 8]

预定义日志level级别

swift 复制代码
func log(level: String, message: String) {
    print("[\(level)] \(message)")
}

// 部分应用:绑定 log level
func logger(for level: String) -> (String) -> Void {
    return { message in log(level: level, message: message) }
}

let errorLog = logger(for: "ERROR")
let infoLog = logger(for: "INFO")

errorLog("Something went wrong")
infoLog("App started")
相关推荐
大熊猫侯佩2 天前
WWDC 25 玻璃态星际联盟:SwiftUI 视图协同“防御协议”
swiftui·swift·wwdc
无知的前端4 天前
一文精通-Combine 框架详解及使用示例
ios·swift
无知的前端4 天前
一文读懂 - Swift 和 Objective-C 创建对象时内存分配机制
ios·性能优化·swift
杂雾无尘4 天前
分享一个让代码更整洁的 Xcode 开发小技巧:设置文件目标平台
ios·swift·apple
桦说编程4 天前
CompletableFuture 的第四种调用模式
java·性能优化·函数式编程
大熊猫侯佩5 天前
WWDC 25 极地冰原撸码危机:InlineArray 与 Span 的绝地反击
swift·apple·wwdc
东坡肘子5 天前
Xcode 26 beta 4,要崩我们一起崩 | 肘子的 Swift 周报 #096
swiftui·swift·apple
杂雾无尘6 天前
解密 Swift 5.5 中的 @MainActor, 深入了解其优势与误区
ios·swift·客户端
胡桃夹夹子7 天前
xcode swift项目运行、连接真机运行报错,引入文件夹失败
cocoa·xcode·swift
卢叁7 天前
关于代码优化的一点思考
ios·swift