kangyana / daily-question

When your heart is set on something, you get closer to your goal with each passing day.
https://www.webpack.top
MIT License
3 stars 0 forks source link

【Q021】函数柯里化 #21

Open kangyana opened 2 years ago

kangyana commented 2 years ago

1. 柯里化(Currying)

柯里化(Currying)是一种关于函数的高阶技术。它不仅被用于 JavaScript,还被用于其他编程语言。 柯里化 是一种函数的 转换 ,它是指将一个函数从可调用的 f(a, b, c) 转换为可调用的 f(a)(b)(c)。 柯里化 不会调用函数,它只是对函数进行转换。

// 柯里化函数
function curry(f) {
  return function(a) {
    return function(b) {
      return f(a, b);
    };
  };
}

// 求和函数
function sum(a, b) {
  return a + b;
}

console.log(curry(sum)(1)(2)) // 3

正如你所看到的,实现非常简单:只有两个包装器(wrapper)。

柯里化更高级的实现,例如 lodash 库的 _.curry,会返回一个包装器,该包装器允许函数被正常调用或者以偏函数(partial)的方式调用:

function sum(a, b) {
  return a + b;
}

var curriedSum = _.curry(sum); // 使用来自 lodash 库的 _.curry
console.log(curriedSum(1, 2)); // 3,仍可正常调用
console.log(curriedSum(1)(2)); // 3,以偏函数的方式调用

2. 柯里化的目的

要了解它的好处,我们需要一个实际中的例子。

例如,我们有一个用于格式化和输出信息的日志(logging)函数 log(date, importance, message)。 在实际项目中,此类函数具有很多有用的功能,例如通过网络发送日志(log):

function log(date, importance, message) {
  console.log(`[${date.getHours()}:${date.getMinutes()}] [${importance}] ${message}`);
}

柯里化之后,log 仍正常运行,也可以以柯里化形式运行:

log = _.curry(log);

log(new Date(), "DEBUG", "some debug"); // log(a, b, c)

log(new Date())("DEBUG")("some debug"); // log(a)(b)(c)

现在可以轻松地为当前日志创建便捷函数:

// logNow 会是带有固定第一个参数的日志的偏函数
var logNow = log(new Date());

// 使用它
logNow("INFO", "message"); // [HH:mm] INFO message

现在,logNow 是具有固定第一个参数的 log,换句话说,就是更简短的“偏应用函数(partially applied function)”或“偏函数(partial)”。

可以更进一步,为当前的调试日志(debug log)提供便捷函数:

var debugNow = logNow("DEBUG");

debugNow("message"); // [HH:mm] DEBUG message

总结:

3. 柯里化的实现

function curry(func) {
  return function curried(...args) {
    if (args.length >= func.length) {
      return func.apply(this, args);
    } else {
      return function(...args2) {
        return curried.apply(this, args.concat(args2));
      }
    }
  };
}

4. 总结