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 天前
ios开发方向——swift并发进阶核心 Task、Actor、await 详解
开发语言·学习·ios·swift
用户79457223954133 天前
【AFNetworking】OC 时代网络请求事实标准,Alamofire 的前身
objective-c·swift
报错小能手3 天前
SwiftUI 框架 认识 SwiftUI 视图结构 + 布局
ui·ios·swift
东坡肘子3 天前
被 Vibe 摧毁的版权壁垒,与开发者的新护城河 -- 肘子的 Swift 周报 #131
人工智能·swiftui·swift
报错小能手4 天前
ios开发方向——swift错误处理:do/try/catch、Result、throws
开发语言·学习·ios·swift
小夏子_riotous4 天前
openstack的使用——5. Swift服务的基本使用
linux·运维·开发语言·分布式·云计算·openstack·swift
mCell4 天前
MacOS 下实现 AI 操控电脑(Computer Use)的思考
macos·agent·swift
用户79457223954134 天前
【DGCharts】iOS 图表渲染事实标准——8 种图表类型、高度可定制,3 行代码画出一条折线
swiftui·swift
chaoguo12344 天前
Any metadata 的内存布局
swift·metadata·value witness table
tangweiguo030519876 天前
SwiftUI布局完全指南:从入门到精通
ios·swift