柯里化是将多参数函数转换为单参数函数序列的技术,通过闭包递归收集参数,支持参数复用、延迟计算与函数组合,可扩展实现占位符机制提升灵活性。

柯里化(Currying)是函数式编程中的一种技术,它将使用多个参数的函数转换成一系列使用单个参数的函数。每次调用只传递一个参数,返回一个新的函数,直到所有参数都传齐后,执行原函数并返回结果。
基本实现方式
实现一个简单的柯里化函数,核心思路是利用闭包和递归收集参数:
<strong>function curry(fn) {<br> return function curried(...args) {<br> if (args.length >= fn.length) {<br> return fn.apply(this, args);<br> } else {<br> return function(...nextArgs) {<br> return curried.apply(this, args.concat(nextArgs));<br> };<br> }<br> };<br>}</strong>
说明:
- fn.length 表示原函数期望的参数个数
- 如果当前传入的参数数量足够,直接执行原函数
- 否则返回一个新函数,继续等待接收剩余参数
使用示例:
立即学习“Java免费学习笔记(深入)”;
<strong>function add(a, b, c) {<br> return a + b + c;<br>}<br><br>const curriedAdd = curry(add);<br><br>console.log(curriedAdd(1)(2)(3)); // 6<br>console.log(curriedAdd(1, 2)(3)); // 6<br>console.log(curriedAdd(1)(2, 3)); // 6</strong>
支持占位符的柯里化
有时希望跳过某个参数稍后填充,可以引入占位符机制(如 lodash 中的 `_`):
<strong>function advancedCurry(fn, placeholder = '_') {<br> return function curried(...args) {<br> const filledArgs = [];<br> let hasPlaceholder = false;<br><br> for (let i = 0; i < args.length; i++) {<br> if (args[i] === placeholder) {<br> hasPlaceholder = true;<br> break;<br> }<br> filledArgs.push(args[i]);<br> }<br><br> if (!hasPlaceholder && filledArgs.length >= fn.length) {<br> return fn.apply(this, filledArgs);<br> }<br><br> return function(...nextArgs) {<br> const combined = [];<br> let nextIndex = 0;<br> for (let i = 0; i < args.length; i++) {<br> if (args[i] === placeholder && nextIndex < nextArgs.length) {<br> combined.push(nextArgs[nextIndex++]);<br> } else if (args[i] !== placeholder) {<br> combined.push(args[i]);<br> }<br> }<br> while (nextIndex < nextArgs.length) {<br> combined.push(nextArgs[nextIndex++]);<br> }<br> return curried.apply(this, combined);<br> };<br> };<br>}</strong>
这种实现允许你写类似 curriedFn(1, '_', 3)(2) 的调用方式。
实际应用场景
-
参数复用:固定部分参数生成新函数,比如日志函数
log(level, msg)可以柯里化出errorLog = curryLog('error') - 延迟计算:不立即执行,等所有参数到位后再运行
- 函数组合:便于与其他高阶函数配合使用,提升代码可读性和模块性
基本上就这些。柯里化让函数更灵活,但也要注意过度使用可能导致调试困难或性能损耗。











