17370845950

javascript中的柯里化是什么_如何应用它?
柯里化是将多参数函数转换为一系列单参数函数的过程,通过闭包累积参数直至满足原函数形参个数才执行;支持提前绑定、函数复用与配置化,适用于日志封装、数组操作及高阶函数构建等场景。

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

柯里化的核心特点

它不是简单地“分次传参”,而是通过闭包保存已传入的参数,逐步累积,最终触发计算。关键在于:参数数量固定、每次只收一个、返回函数而非结果。

  • 原函数 f(a, b, c) 经柯里化后变成 f(a)(b)(c)
  • 中间调用如 f(a)(b) 不执行逻辑,只返回等待 c 的函数
  • 支持“提前绑定”部分参数,实现函数复用和配置化

手写一个通用柯里化函数

可以借助函数的 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));
      };
    }
  };
}

用法示例:

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

实际中怎么用?几个典型场景

柯里化不是炫技,它在简化配置、封装逻辑、提升可读性上很实用。

  • 创建预设行为的函数:比如日志函数 log(level, message)const error = curry(log)('ERROR'),后续只需传 message
  • 与数组方法配合:如 [1,2,3].map(curry(Math.pow)(2)) 得到平方数组 [1,4,9]
  • 替代 bind 做参数绑定:比 fn.bind(null, a) 更灵活,支持链式传参且不依赖 this
  • 构建高阶工具函数:如 const match = curry((regex, str) => str.match(regex)),再写 const hasNumber = match(/\d+/)

注意边界情况

柯里化适合参数明确、顺序固定的函数。遇到以下情况要谨慎:

  • 原函数有默认参数或 rest 参数时,fn.length 可能不准,需手动指定参数个数
  • 箭头函数没有 argumentslength,无法直接 curry,需转为普通函数
  • 频繁柯里化可能增加调用开销,性能敏感场景建议预编译或用闭包手动写

基本上就这些。柯里化本质是函数式编程中“分解与组合”的体现,用得好能让代码更清晰、更可复用,但别为了柯里化而柯里化。