柯里化是将多参数函数转换为一系列单参数函数链的过程,核心是参数复用与延迟执行,通过闭包保存已传参数,满足形参个数时执行,支持分步调用与跨场景复用。

柯里化(Currying)是把一个接收多个参数的函数,转换成一系列只接收一个参数的函数链的过程。它的核心目的是参数复用和提前绑定部分参数,让函数更灵活、可组合、易测试。
柯里化的本质:固定参数,延迟执行
比如原函数 add(a, b, c),柯里化后变成 add(a)(b)(c)。每次调用只传一个参数,返回新函数,直到参数数量满足才真正执行。中间每一步都“记住”了已传的参数——这就是参数复用的基础。
- 不是简单地把多参变单参,而是通过闭包保存已传参数
- 最终调用时,所有累积的参数一起参与运算
- 支持不一次性传完,可以分多次、跨时间、跨模块复用同一组前置参数
手写一个通用柯里化函数
关键点:记录期望参数个数(通过 fn.length)、用闭包暂存已传参数、递归返回新函数、满足数量时立即执行。
function curry(fn) {
return function curried(...args) {
if (args.length >= fn.length) {
return fn.apply(this, args);
} else {
return function(...nextArgs) {
return curried.apply(this, args.concat(nextArgs));
};
}
};
}使用示例:
立即学习“Java免费学习笔记(深入)”;
function multiply(a, b, c) {
return a * b * c;
}
const curriedMultiply = curry(multiply);
console.log(curriedMultiply(2)(3)(4)); // 24
console.log(curriedMultiply(2, 3)(4)); // 24
console.log(curriedMultiply(2)(3, 4)); // 24参数复用的实际场景
柯里化真正价值体现在需要反复使用相同配置的场景中:
-
日志封装:固定日志级别,如
const errorLog = curry(console.log)('ERROR'),后续只需传消息 -
API 请求预设:固定 base URL 或 token,
const requestUser = curry(fetchWithAuth)('https://api.com')('Bearer xxx') -
事件处理器配置:如
const handleClick = curry(handleAction)('save')('form-1'),生成特定行为的回调 -
函数式编程组合:与
map、filter配合,例如[1,2,3].map(curry(Math.pow)(2))得到平方数组
注意边界与替代方案
柯里化不是万能的。它依赖函数形参长度,对 rest 参数(...args)或动态参数处理较弱;过度柯里化会增加理解成本。现代开发中也可考虑:
- 显式工厂函数:
const makeAdder = x => y => x + y(更直观、无反射开销) - Lodash 的
_.curry(支持占位符、自定义参数长度) - TypeScript 中结合泛型做类型推导,提升安全性
不复杂但容易忽略:柯里化是思维转变——从“一次调用完成任务”,到“逐步构造专属函数”。











