17370845950

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

柯里化(Currying)是函数式编程中的一种技术,它将使用多个参数的函数转换成一系列使用单个参数的函数。每次调用只传递一个参数,返回一个新的函数,直到所有参数都传齐后,执行原函数并返回结果。

基本实现方式

实现一个简单的柯里化函数,核心思路是利用闭包和递归收集参数:

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));
};
}
};
}

说明:

  • fn.length 表示原函数期望的参数个数
  • 如果当前传入的参数数量足够,直接执行原函数
  • 否则返回一个新函数,继续等待接收剩余参数

使用示例:

function add(a, b, c) {
return 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

支持占位符的柯里化

有时希望跳过某个参数稍后填充,可以引入占位符机制(如 lodash 中的 `_`):

function advancedCurry(fn, placeholder = '_') {
return function curried(...args) {
const filledArgs = [];
let hasPlaceholder = false;

for (let i = 0; i < args.length; i++) {
if (args[i] === placeholder) {
hasPlaceholder = true;
break;
}
filledArgs.push(args[i]);
}

if (!hasPlaceholder && filledArgs.length >= fn.length) {
return fn.apply(this, filledArgs);
}

return function(...nextArgs) {
const combined = [];
let nextIndex = 0;
for (let i = 0; i < args.length; i++) {
if (args[i] === placeholder && nextIndex < nextArgs.length) {
combined.push(nextArgs[nextIndex++]);
} else if (args[i] !== placeholder) {
combined.push(args[i]);
}
}
while (nextIndex < nextArgs.length) {
combined.push(nextArgs[nextIndex++]);
}
return curried.apply(this, combined);
};
};
}

这种实现允许你写类似 curriedFn(1, '_', 3)(2) 的调用方式。

实际应用场景

  • 参数复用:固定部分参数生成新函数,比如日志函数 log(level, msg) 可以柯里化出 errorLog = curryLog('error')
  • 延迟计算:不立即执行,等所有参数到位后再运行
  • 函数组合:便于与其他高阶函数配合使用,提升代码可读性和模块性

基本上就这些。柯里化让函数更灵活,但也要注意过度使用可能导致调试困难或性能损耗。