函数的柯里化

文章介绍了函数柯里化的基本概念,通过示例展示了如何将多参数函数转换为一系列单参数函数。柯里化可以减少代码冗余,提高可读性,并在实际应用中,如列表数据处理和优化计算性能(如Memoization)中发挥作用。文章提供了柯里化函数的实现方式,并通过map方法的例子解释了柯里化在处理数组数据时的便利性。
摘要由CSDN通过智能技术生成

函数的柯里化

在数学和计算机科学中,柯里化是一种将使用多个参数的一个函数转换成一系列使用一个参数的函数的技术。

currying 又称部分求值。一个 currying 的函数首先会接受一些参数,接受了这些参数之后,该函数不会立即求值,而是继续返回另外一个函数,刚才传入的参数在函数形成的闭包中被保存起来。待到函数被真正求值的时候,之前传入的所有参数都会被一次性用于求值。
  currying 简单的说就是:只传递给函数一部分参数来调用它,让它返回一个函数去处理剩下的参数。你可以一次性地调用 curry 函数,也可以每次只传一个参数分多次调用。

function add(a, b, c, d) {
  // add.length 返回参数个数
  return a + b + c + d;
}

//   需要固定参数的函数
function fixedParamsCurry(fn) {
  //arguments = [add, 1, 2]
  var _arg = [].slice.call(arguments, 1);
  return function () {
    //arguments = [2, 3]
    var newArg = _arg.concat([].slice.call(arguments, 0));
    return fn.apply(this, newArg);
  };
}
//   var newAdd = fixedParamsCurry(add, 1, 2);
//   console.log(newAdd(2, 3));

function Curry(fn, length) {
  // 没有传入length参数 为false
  var length = length || fn.length;
  return function () {
    //如果传入参数个数小于要参数个数
    if (arguments.length < length) {
      // 将传入函数和参数合并数组
      //fn.concat([1]) = [fn, 1]
      var combined = [fn].concat([].slice.call(arguments, 0));
      //返回递归一个固化函数
      return Curry(
        fixedParamsCurry.apply(this, combined),
        length - arguments.length
      );
    } else {
      // 递归出口
      return fn.apply(this, arguments);
    }
  };
}

var newAdd = Curry(add);
var num = newAdd(1)(2)(3)(4); //10

最近,朋友 T 在准备面试,他为一道编程题所困,向我求助。原题如下:

// 写一个 sum 方法,当使用下面的语法调用时,能正常工作
console.log(sum(2, 3)); // Outputs 5
console.log(sum(2)(3)); // Outputs 5
这道题要考察的,就是对函数柯里化的理解。让我们先来解析一下题目的要求:

如果传递两个参数,我们只需将它们相加并返回。
否则,我们假设它是以 sum(2)(3)的形式被调用的,所以我们返回一个匿名函数,它将传递给 sum()(在本例中为 2)的参数和传递给匿名函数的参数(在本例中为 3)。
所以,sum 函数可以这样写:

function sum(x) {
  if (arguments.length == 2) {
    return arguments[0] + arguments[1];
  }

  return function (y) {
    return x + y;
  };
}

arguments 的用法挺灵活的,在这里它则用于分割两种不同的情况。当参数只有一个的时候,进行柯里化的处理。

那么,到底什么是函数的柯里化呢?接下来,我们将从概念出发,探究函数柯里化的实现与用途。

什么是柯里化

柯里化,是函数式编程的一个重要概念。它既能减少代码冗余,也能增加可读性。另外,附带着还能用来装逼。

先给出柯里化的定义:在数学和计算机科学中,柯里化是一种将使用多个参数的一个函数转换成一系列使用一个参数的函数的技术。

柯里化的定义,理解起来有点费劲。为了更好地理解,先看下面这个例子:

function sum(a, b, c) {
  console.log(a + b + c);
}
sum(1, 2, 3); // 6

毫无疑问,sum 是个简单的累加函数,接受 3 个参数,输出累加的结果。

假设有这样的需求,sum 的前 2 个参数保持不变,最后一个参数可以随意。那么就会想到,在函数内,是否可以把前 2 个参数的相加过程,给抽离出来,因为参数都是相同的,没必要每次都做运算。

如果先不管函数内的具体实现,调用的写法可以是这样: sum(1, 2)(3); 或这样 sum(1, 2)(10); 。就是,先把前 2 个参数的运算结果拿到后,再与第 3 个参数相加。

这其实就是函数柯里化的简单应用。

柯里化的实现

sum(1, 2)(3); 这样的写法,并不常见。拆开来看,sum(1, 2) 返回的应该还是个函数,因为后面还有 (3) 需要执行。

那么反过来,从最后一个参数,从右往左看,它的左侧必然是一个函数。以此类推,如果前面有 n 个(),那就是有 n 个函数返回了结果,只是返回的结果,还是一个函数。是不是有点递归的意思?

** 网上有一些不同的柯里化的实现方式,以下是个人觉得最容易理解的写法:**

function curry(fn, currArgs) {
  return function () {
    let args = [].slice.call(arguments);

    // 首次调用时,若未提供最后一个参数currArgs,则不用进行args的拼接
    if (currArgs !== undefined) {
      args = args.concat(currArgs);
    }

    // 递归调用
    if (args.length < fn.length) {
      return curry(fn, args);
    }

    // 递归出口
    return fn.apply(null, args);
  };
}

解析一下 curry 函数的写法:

首先,它有 2 个参数,fn 指的就是本文一开始的源处理函数 sum。currArgs 是调用 curry 时传入的参数列表,比如 (1, 2)(3) 这样的。

再看到 curry 函数内部,它会整个返回一个匿名函数。

再接下来的 let args = [].slice.call(arguments);,意思是将 arguments 数组化。arguments 是一个类数组的结构,它并不是一个真的数组,所以没法使用数组的方法。我们用了 call 的方法,就能愉快地对 args 使用数组的原生方法了。在这篇 「干货」细说 call、apply 以及 bind 的区别和用法 中,有关于 call 更详细的用法介绍。

currArgs !== undefined 的判断,是为了解决递归调用时的参数拼接。

最后,判断 args 的个数,是否与 fn (也就是 sum )的参数个数相等,相等了就可以把参数都传给 fn,进行输出;否则,继续递归调用,直到两者相等。

测试一下:

function sum(a, b, c) {
  console.log(a + b + c);
}

const fn = curry(sum);

fn(1, 2, 3); // 6
fn(1, 2)(3); // 6
fn(1)(2, 3); // 6
fn(1)(2)(3); // 6

都能输出 6 了,搞定!

柯里化的用途
理解了柯里化的实现之后,让我们来看一下它的实际应用。柯里化的目的是,减少代码冗余,以及增加代码的可读性。来看下面这个例子:

const persons = [
  { name: "kevin", age: 4 },
  { name: "bob", age: 5 },
];

// 这里的 curry 函数,之前已实现
const getProp = curry(function (obj, index) {
  const args = [].slice.call(arguments);
  return obj[args[args.length - 1]];
});

const ages = persons.map(getProp("age")); // [4, 5]
const names = persons.map(getProp("name")); // ['kevin', 'bob']

在实际的业务中,我们常会遇到类似的列表数据。用 getProp 就可以很方便地,取出列表中某个 key 对应的值。

需要注意的是,const names = persons.map(getProp(‘name’)); 执行这条语句时 getProp 的参数只有一个 name,而定义 getProp 方法时,传入 curry 的参数有 2 个,obj 和 index(这里必须写 2 个及以上的参数)。

为什么要这么写?关键就在于 arguments 的隐式传参。

const getProp = curry(function (obj, index) {
  console.log(arguments);
  // 会输出 4 个类数组,取其中一个来看
  // {
  // 0: {name: "kevin", age: 4},
  // 1: 0,
  // 2: [
  // {name: "kevin", age: 4},
  // {name: "bob", age: 5}
  // ],
  // 3: "age"
  // }
});

map 是 Array 的原生方法,它的用法如下:

var new_array = arr.map(function callback(currentValue[, index[, array]]) {
// Return element for new_array
}[, thisArg]);

所以,我们传入的 name,就排在了 arguments 的最后。为了拿到 name 对应的值,需要对类数组 arguments 做点转换,让它可以使用 Array 的原生方法。所以,最终 getProp 方法定义成了这样:

const getProp = curry(function (obj, index) {
  const args = [].slice.call(arguments);
  return obj[args[args.length - 1]];
});

当然,还有另外一种写法,curry 的实现更好理解,但是调用的代码却变多了,大家可以根据实际情况进行取舍。

const getProp = curry(function (key, obj) {
  return obj[key];
});

const ages = persons.map((item) => {
  return getProp(item)("age");
});
const names = persons.map((item) => {
  return getProp(item)("name");
});

最后,来看一个 Memoization 的例子。它用于优化比较耗时的计算,通过将计算结果缓存到内存中,这样对于同样的输入值,下次只需要中内存中读取结果。

function memoizeFunction(func) {
  const cache = {};
  return function () {
    let key = arguments[0];
    if (cache[key]) {
      return cache[key];
    } else {
      const val = func.apply(null, arguments);
      cache[key] = val;
      return val;
    }
  };
}

const fibonacci = memoizeFunction(function (n) {
  return n === 0 || n === 1 ? n : fibonacci(n - 1) + fibonacci(n - 2);
});

console.log(fibonacci(100)); // 输出354224848179262000000
console.log(fibonacci(100)); // 输出354224848179262000000

代码中,第 2 次计算 fibonacci(100) 则只需要在内存中直接读取结果。

总结
函数的柯里化,是 Javascript 中函数式编程的一个重要概念。它返回的,是一个函数的函数。其实现方式,需要依赖参数以及递归,通过拆分参数的方式,来调用一个多参数的函数方法,以达到减少代码冗余,增加可读性的目的。

虽然一开始理解起来有点云里雾里的,但一旦理解了其中的含义和具体的使用场景,用起来就会得心应手了。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值