什么是柯里化(currying)?

维基百科中的解释是:柯里化是把接受多个参数的函数变换成接受一个单一参数(最初函数的第一个参数)的函数,并且返回接受余下的参数而且返回结果的新函数的技术。意思就是当函数被调用时,返回的函数还需要设置一些传入的参数。

首先来看一个简单的例子,有下面一个函数:

function add(num1, num2) {
return num1 + num2;
}

我们把它改写成下面这样:

var fn = function(a) {
return function (b) {
return a + b;
}
}

可以这样调用函数:fn(2)(3)。上面使用了匿名函数来实现多参数函数的方法,虽然这并不是柯里化的函数,但可以帮助我们理解柯里化的含义。

实现通用柯里化函数###

我们可以在内置构造函数Function()的原型上来添加一个柯里化函数,这样所有的函数都可以调用。下面是通用柯里化函数的实现:

Function.prototype.currying = function () {
var that = this;
var args = [].slice.call(arguments);
return function () {
return that.apply(null, args.concat([].slice.call(arguments)));
}
}

现在用柯里化函数将上面的add函数柯里化:

var curriedAdd = add.currying(2);
curriedAdd(3); // 5

也可以一次性传入两个参数:

var curriedAdd = add.currying(2, 3);
curriedAdd(); // 5

我们知道在原生对象的原型上扩展方法是不太好的,因为可能会导致命名冲突。所以最好不要把currying函数扩展在Function的原型上,下面是改写的currying函数:

function currying(fn) {
var args = [].slice.call(arguments, 1);
return function () {
return that.apply(null, args.concat([].slice.call(arguments)));
}
}

改写之后currying函数的第一个参数是要被柯里化的函数,可以这样调用:

var curriedAdd = currying(add, 2);
curriedAdd(3); // 5

var curriedAdd = currying(add, 2, 3);
curriedAdd(); // 5

上面的add函数只是两个数字的相加,如果我们需要n个数字相加,上面的currying函数已经不能满足要求了,下面是修改后的currying函数:

function currying(fn) {
var argsArr = [];
return function () {
if (arguments.length === 0) {
return fn.apply(null, argsArr);
} else {
[].push.apply(argsArr, arguments);
}
}
}

多个数字相加:

var add = function () {
var num = 0;
[].forEach.call(arguments, function (item, i) {
num += item;
})
return num;
} var curriedAdd = currying(add);
curriedAdd(2);
curriedAdd(3);
curriedAdd(4);
curriedAdd(5);
curriedAdd(); // 14

这样做有什么好处呢?假如说我们只想知道这个月花了多少钱,而中间的某一天之前花了多少我们并不想知道,我们只在乎结果,不在乎过程,上面的currying函数很好地解决了这个问题。有的人说这样做可以节省性能,我倒觉得这和性能没多大关系,或者说这样做的目的并不是为了性能,因为每次计算结果和最后一起计算结果是一样的,都是要计算一样的次数。还有一个好处就是可以复用currying函数,比如我们要多个数字相乘或者其他操作,都可以用currying函数,处理数字只需修改fn参数就可以。

说到柯里化就不得不说Function.prototype.bind这个方法了,它也实现了函数的柯里化。我们可以自己来实现一个bind函数:

function bind(fn, context) {
var args = [].slice.call(arguments, 2);
return function () {
return fn.apply(context, args.concat([].slice.call(arguments)));
}
}

假如我们需要改变fn中的this上下文,就可以用bind函数,否则可以用currying函数。

JavaScript之函数柯里化的更多相关文章

  1. 简单粗暴详细讲解javascript实现函数柯里化与反柯里化

    函数柯里化(黑人问号脸)???Currying(黑人问号脸)???妥妥的中式翻译既视感:下面来一起看看究竟什么是函数柯里化: 维基百科的解释是:把接收多个参数的函数变换成接收一个单一参数(最初函数的第 ...

  2. 简单粗暴详细讲解javascript实现函数柯里化

    函数柯里化(黑人问号脸)???Currying(黑人问号脸)???妥妥的中式翻译既视感:下面来一起看看究竟什么是函数柯里化: 维基百科的解释是:把接收多个参数的函数变换成接收一个单一参数(最初函数的第 ...

  3. Javascript函数柯里化(curry)

    函数柯里化currying,是函数式编程非常重要的一个标志.它的实现需要满足以下条件,首先就是函数可以作为参数进行传递,然后就是函数可以作为返回值return出去.我们依靠这个特性编写很多优雅酷炫的代 ...

  4. 深入理解javascript函数进阶系列第二篇——函数柯里化

    前面的话 函数柯里化currying的概念最早由俄国数学家Moses Schönfinkel发明,而后由著名的数理逻辑学家Haskell Curry将其丰富和发展,currying由此得名.本文将详细 ...

  5. JavaScript中的事件循环机制跟函数柯里化

    一.事件循环机制的理解 test();//按秒输出5个5 function test() { for (var i = 0; i < 5; i++) { setTimeout(() => ...

  6. JavaScript函数柯里化的一些思考

    1. 高阶函数的坑 在学习柯里化之前,我们首先来看下面一段代码: var f1 = function(x){ return f(x); }; f1(x); 很多同学都能看出来,这些写是非常傻的,因为函 ...

  7. 一道javascript面试题(闭包与函数柯里化)

    要求写一个函数add(),分别实现能如下效果: (1)console.log(add(1)(2)(3)(4)()); (2)console.log(add(1,2)(3,4)()); (3)conso ...

  8. 精读JavaScript模式(六),Memoization模式与函数柯里化的应用

    假期就这么结束了!十天假就有三天在路上,真的难受!想想假期除了看了两场电影貌似也没做什么深刻印象的事情.流浪地球,特效还是很赞,不过对于感情的描写还是逃不掉拖沓和尴尬的通病,对于国产科幻还是抱有支持的 ...

  9. JavaScript中的函数柯里化与反柯里化

    一.柯里化定义 在计算机科学中,柯里化是把 接受多个参数的函数 变换成 接受一个单一参数(最初函数的第一个参数)的函数 并且返回 接受余下参数且返回结果的新函数的技术 高阶函数 高阶函数是实现柯里化的 ...

随机推荐

  1. iOS之Custom UIViewController Transition

    本文学习下自定义ViewController的切换,从无交互的到交互式切换. (本文已同步到我的小站:icocoa,欢迎访问.) iOS7中定义了3个协议: UIViewControllerTrans ...

  2. iOS之面试题:阿里-P6一面-参考思路

    阿里-p6-一面  1.介绍下内存的几大区域? 2.你是如何组件化解耦的? 3.runtime如何通过selector找到对应的IMP地址 4.runloop内部实现逻辑? 5.你理解的多线程? 6. ...

  3. duplicate symbol _OBJC_IVAR

    duplicate symbol _OBJC_IVAR - Kingdev - 博客园 代码合并问题 git合并提示冲突文件为project.pbxproj,先去掉所有冲突提示<<< ...

  4. 手机版的百度map封装,使用gps定位

    代码如下,包自己引 包参考 一个百度MAP导航的基础封装 使用的是浏览器调用gps定位 修改了标注的大小 效果如图: 代码...... <!DOCTYPE html> <html&g ...

  5. js 时间转换毫秒的四种方法(转)

    将时间转换为毫秒数的方法有四个: Date.parse()Date.UTCvalueOf()getTime() 1. Date.parse():该方法接受一个表示日期的字符串参数,然后尝试根据这个日期 ...

  6. 各种快速幂(qaq)

    今天分享下各种快速幂(有点坑),首先说一下快速幂的原理, 以下以求a的b次方来介绍 [1]  把b转换成二进制数. 该二进制数第i位的权为   例如 11的二进制是1011 11 = 2³×1 + 2 ...

  7. 偏前端 + rsa加解密 + jsencrypt.min.js--(新增超长字符分段加解密)

    <html> <head> <title>JavaScript RSA Encryption</title> <meta charset=&quo ...

  8. python学习笔记:第9天 函数初步

    1. 函数的定义及调用 函数:所谓的函数可以看作是对一段代码的封装,也是对一个功能模块的封装,这样方便在下次想用这个功能的时候直接调用这个功能模块,而不用重新去写. 函数的定义:我们使用def关键字来 ...

  9. 【Mac】gem install 出错 You don't have write permissions for the /Library/Ruby/Gems

    问题描述 RedisDump 是一个用于 Redis 数据导人/导出的工具,是基于 Ruby 实现的,需要先安装 Ruby.但因为 Mac 自带有 Ruby 所以我直接用gem install red ...

  10. C语言之数组和函数

    数组 一维数组 定义:类型符 数组名 [常量表达式]  int a[]={1,2,3,4,5,}; #include<stdio.h> #define NUM 6 void main() ...