柯里化是将多参数函数转换为一系列单参数函数的过程,通过闭包累积参数、延迟执行,支持灵活调用与占位符填充,常用于预设配置、函数组合和API封装。

柯里化(Currying)是将一个接收多个参数的函数,转换为一系列只接收一个参数的函数的过程。每次调用返回一个新的函数,直到所有参数都传齐,才真正执行原函数。
柯里化的核心特点
它不是简单地“分多次传参”,而是通过闭包保存已传入的参数,逐步累积,最终触发计算。关键在于:参数个数确定、延迟执行、返回函数链。
手动实现一个基础柯里化函数
假设我们要柯里化一个加法函数 add(a, b, c),目标是支持 add(1)(2)(3) 或 add(1, 2)(3) 等灵活调用方式。一个健壮的实现需判断参数是否已满足原函数要求:
- 用
func.length获取原函数期望的形参个数 - 每次调用收集实参,若数量不足,返回新柯里化函数;若足够,则立即执行
- 利用闭包保留已传参数(用扩展运算符
[...args]合并)
示例代码:
立即学习“Java免费学习笔记(深入)”;
function curry(func) {
return function curried(...args) {
if (args.length >= func.length) {
return func.apply(this, args);
} else {
return function(...moreArgs) {
return curried.apply(this, [...args, ...moreArgs]);
};
}
};
}
// 使用
const add = (a, b, c) => a + b + c;
const curriedAdd = curry(add);
console.log(curriedAdd(1)(2)(3)); // 6
console.log(curriedAdd(1, 2)(3)); // 6
console.log(curriedAdd(1)(2, 3)); // 6
支持占位符的增强版柯里化
真实开发中常需跳过某些参数,比如 curriedFn(1, _, 3)(2) 表示把第二个位置留空,最后传入 2 填上。这时需定义一个占位符(如 _),并在合并参数时跳过它:
- 预先定义唯一占位符:
const placeholder = Symbol('placeholder') - 合并参数时,用占位符代替未传位置,后续调用再填充
- 判断是否“满员”时,按非占位符的实际参数个数计算
这会让柯里化更贴近 Lodash 的 _.curry 行为,适合复杂场景复用。
柯里化的典型用途
它本身不解决性能问题,但能提升函数组合性与可配置性:
-
预设配置:如
const sendToApi = curry(fetchWithAuth)(token),生成带认证信息的专用函数 -
函数式编程组合:配合
map、filter时避免写匿名函数,例如arr.map(curry(Math.pow)(2))表示平方 - API 封装降噪:把多参数 SDK 方法柯里化后,业务层只需传业务参数,环境/配置由上层固化
注意:过度柯里化会增加理解成本,应权衡可读性与灵活性。











