柯里化是将多参数函数转换为一系列单参数函数的技术。它通过闭包保存已传参数,参数足够时执行原函数,不足时返回新函数继续接收;适用于预设配置、事件处理、函数式组合及API封装等场景。

什么是柯里化:把多参数函数拆成一连串单参数函数
柯里化(Currying)不是“把函数变复杂”,而是把一个接收多个参数的函数,转换为一系列每次只接收一个参数的函数。每次调用返回一个新的函数,直到参数个数满足原函数要求,才真正执行逻辑。
手写一个通用柯里化函数
核心思路是:保存已传入的参数,用闭包维持状态;当累积参数够了,就调用原函数;不够就返回新函数继续接收。
简单可运行版本如下:
function curry(fn) {
return function curried(...args) {
if (args.length >= fn.length) {
return fn.apply(this, args);
} else {
return function(...moreArgs) {
return curried.apply(this, args.concat(moreArgs));
};
}
};
}
使用示例:
立即学习“Java免费学习笔记(深入)”;
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
实际中真正用得上的场景
柯里化不是炫技,它在以下场景能显著提升代码清晰度和复用性:
-
预设配置或上下文:比如日志函数
log(level, message),可柯里化为const errorLog = curry(log)('ERROR'),后续直接errorLog('user not found'),避免重复传 level。 -
事件处理器中固定部分参数:如
handleClick(id, event),绑定时写onClick={curry(handleClick)(item.id)},比箭头函数() => handleClick(item.id, event)更轻量,也避免每次渲染都新建函数(利于 React 性能)。 -
函数式组合与管道(pipeline):配合
map、filter等高阶函数时,柯里化让参数顺序更自然。例如['1', '2', '3'].map(curry(parseInt)(10))直接转为十进制数字,比map(s => parseInt(s, 10))更简洁且无闭包变量依赖。 -
API 封装与适配:对接第三方 SDK(如 axios 的
get(url, config)),可柯里化出const apiGet = curry(axios.get)({ timeout: 5000 }),后续调用只需传 url,统一超时配置不散落各处。
注意几个常见误区
柯里化 ≠ 偏函数(Partial Application)。偏函数是固定任意个参数(不一定要从左到右),而标准柯里化严格按顺序、每次只收一个。另外,ES6 箭头函数写法虽简洁,但无法通过 fn.length 获取形参个数(因箭头函数没有 arguments 和 length 绑定),所以通用 curry 函数建议用于普通 function 声明/表达式。
不复杂但容易忽略:柯里化本质是控制参数流动节奏,关键不在“怎么写”,而在“什么时候值得拆”——当你发现同一组参数反复传、或者想把“变化的”和“不变的”明确分开时,它就该出场了。











