17370845950

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

柯里化(Currying)是将一个接收多个参数的函数,转换为一系列只接收一个参数的函数的过程。每次调用返回一个新的函数,直到所有参数都传齐,才真正执行原函数。

柯里化的核心特点

它不是简单地“分多次传参”,而是通过闭包保存已传入的参数,逐步累积,最终触发计算。关键在于:参数个数确定、延迟执行、返回函数链。

手动实现一个基础柯里化函数

假设我们要柯里化一个加法函数 add(a, b, c),目标是支持 add(1)(2)(3)add(1, 2)(3) 等灵活调用方式。一个健壮的实现需判断参数是否已满足原函数要求:

  • func.length 获取原函数期望的形参个数
  • 每次调用收集实参,若数量不足,返回新柯里化函数;若足够,则立即执行
  • 利用闭包保留已传参数(用扩展运算符 [...args] 合并)

示例代码:

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),生成带认证信息的专用函数
  • 函数式编程组合:配合 mapfilter 时避免写匿名函数,例如 arr.map(curry(Math.pow)(2)) 表示平方
  • API 封装降噪:把多参数 SDK 方法柯里化后,业务层只需传业务参数,环境/配置由上层固化

注意:过度柯里化会增加理解成本,应权衡可读性与灵活性。