详解JS中的柯里化(currying)


Posted in Javascript onAugust 17, 2017

何为Curry化/柯里化?

curry化来源与数学家 Haskell Curry的名字 (编程语言 Haskell也是以他的名字命名)。

柯里化通常也称部分求值,其含义是给函数分步传递参数,每次传递参数后部分应用参数,并返回一个更具体的函数接受剩下的参数,这中间可嵌套多层这样的接受部分参数函数,直至返回最后结果。

因此柯里化的过程是逐步传参,逐步缩小函数的适用范围,逐步求解的过程。

柯里化一个求和函数

按照分步求值,我们看一个简单的例子

var concat3Words = function (a, b, c) {
 return a+b+c;
};
var concat3WordsCurrying = function(a) {
 return function (b) {
  return function (c) {
   return a+b+c;
  };
 };
};
console.log(concat3Words("foo ","bar ","baza"));   // foo bar baza
console.log(concat3WordsCurrying("foo "));     // [Function]
console.log(concat3WordsCurrying("foo ")("bar ")("baza")); // foo bar baza

可以看到, concat3WordsCurrying("foo ") 是一个 Function,每次调用都返回一个新的函数,该函数接受另一个调用,然后又返回一个新的函数,直至最后返回结果,分布求解,层层递进。(PS:这里利用了闭包的特点)

那么现在我们更进一步,如果要求可传递的参数不止3个,可以传任意多个参数,当不传参数时输出结果?

首先来个普通的实现:

var add = function(items){
 return items.reduce(function(a,b){
  return a+b
 });
};
console.log(add([1,2,3,4]));

但如果要求把每个数乘以10之后再相加,那么:

var add = function (items,multi) {
 return items.map(function (item) {
  return item*multi;
 }).reduce(function (a, b) {
  return a + b
 });
};
console.log(add([1, 2, 3, 4],10));

好在有 map 和 reduce 函数,假如按照这个模式,现在要把每项加1,再汇总,那么我们需要更换map中的函数。

下面看一下柯里化实现:

var adder = function () {
 var _args = [];
 return function () {
  if (arguments.length === 0) {
   return _args.reduce(function (a, b) {
    return a + b;
   });
  }
  [].push.apply(_args, [].slice.call(arguments));
  return arguments.callee;
 }
}; 
var sum = adder();
console.log(sum);  // Function
sum(100,200)(300); // 调用形式灵活,一次调用可输入一个或者多个参数,并且支持链式调用
sum(400);
console.log(sum()); // 1000 (加总计算)

上面 adder是柯里化了的函数,它返回一个新的函数,新的函数接收可分批次接受新的参数,延迟到最后一次计算。

通用的柯里化函数

更典型的柯里化会把最后一次的计算封装进一个函数中,再把这个函数作为参数传入柯里化函数,这样即清晰,又灵活。

例如 每项乘以10, 我们可以把处理函数作为参数传入:

var currying = function (fn) {
 var _args = [];
 return function () {
  if (arguments.length === 0) {
   return fn.apply(this, _args);
  }
  Array.prototype.push.apply(_args, [].slice.call(arguments));
  return arguments.callee;
 }
};
var multi=function () {
 var total = 0;
 for (var i = 0, c; c = arguments[i++];) {
  total += c;
 }
 return total;
};
var sum = currying(multi); 
sum(100,200)(300);
sum(400);
console.log(sum());  // 1000 (空白调用时才真正计算)

这样 sum = currying(multi),调用非常清晰,使用效果也堪称绚丽,例如要累加多个值,可以把多个值作为做个参数 sum(1,2,3),也可以支持链式的调用,sum(1)(2)(3)

柯里化的作用

  • 延迟计算。上面的例子已经比较好低说明了。
  • 参数复用。当在多次调用同一个函数,并且传递的参数绝大多数是相同的,那么该函数可能是一个很好的柯里化候选。
  • 动态创建函数。

这可以是在部分计算出结果后,在此基础上动态生成新的函数处理后面的业务,这样省略了重复计算。或者可以通过将要传入调用函数的参数子集,部分应用到函数中,从而动态创造出一个新函数,这个新函数保存了重复传入的参数(以后不必每次都传)。例如,事件浏览器添加事件的辅助方法:

var addEvent = function(el, type, fn, capture) {
  if (window.addEventListener) {
   el.addEventListener(type, function(e) {
    fn.call(el, e);
   }, capture);
  } else if (window.attachEvent) {
   el.attachEvent("on" + type, function(e) {
    fn.call(el, e);
   });
  } 
 };

每次添加事件处理都要执行一遍 if...else...,其实在一个浏览器中只要一次判定就可以了,把根据一次判定之后的结果动态生成新的函数,以后就不必重新计算。

var addEvent = (function(){
 if (window.addEventListener) {
  return function(el, sType, fn, capture) {
   el.addEventListener(sType, function(e) {
    fn.call(el, e);
   }, (capture));
  };
 } else if (window.attachEvent) {
  return function(el, sType, fn, capture) {
   el.attachEvent("on" + sType, function(e) {
    fn.call(el, e);
   });
  };
 }
})();

这个例子,第一次 if...else... 判断之后,完成了部分计算,动态创建新的函数来处理后面传入的参数,这是一个典型的柯里化。

Function.prototype.bind 方法也是柯里化应用

与 call/apply 方法直接执行不同,bind 方法 将第一个参数设置为函数执行的上下文,其他参数依次传递给调用方法(函数的主体本身不执行,可以看成是延迟执行),并动态创建返回一个新的函数, 这符合柯里化特点。

var foo = {x: 888};
var bar = function () {
 console.log(this.x);
}.bind(foo);    // 绑定
bar(); 
// 888

与 call/apply 方法直接执行不同,bind 方法 将第一个参数设置为函数执行的上下文,其他参数依次传递给调用方法(函数的主体本身不执行,可以看成是延迟执行),并动态创建返回一个新的函数, 这符合柯里化特点。

var foo = {x: 888};
var bar = function () {
 console.log(this.x);
}.bind(foo);    // 绑定
bar(); 
// 888

下面是一个 bind 函数的模拟,testBind 创建并返回新的函数,在新的函数中将真正要执行业务的函数绑定到实参传入的上下文,延迟执行了。

Function.prototype.testBind = function (scope) {
 var fn = this;     //// this 指向的是调用 testBind 方法的一个函数, 
 return function () {
  return fn.apply(scope);
 }
};
var testBindBar = bar.testBind(foo); // 绑定 foo,延迟执行
console.log(testBindBar);    // Function (可见,bind之后返回的是一个延迟执行的新函数)
testBindBar();

这里要注意 prototype 中 this 的理解。

实例

实例1:

var currying = function(fn) {
 // fn 指官员消化老婆的手段
 var args = [].slice.call(arguments, 1);
 // args 指的是那个合法老婆
 return function() {
  // 已经有的老婆和新搞定的老婆们合成一体,方便控制
  var newArgs = args.concat([].slice.call(arguments));
  // 这些老婆们用 fn 这个手段消化利用,完成韦小宝前辈的壮举并返回
  return fn.apply(null, newArgs);
 };
};
// 下为官员如何搞定7个老婆的测试
// 获得合法老婆
var getWife = currying(function() {
 var allWife = [].slice.call(arguments);
 // allwife 就是所有的老婆的,包括暗渡陈仓进来的老婆
 console.log(allWife.join(";"));
}, "合法老婆");
// 获得其他6个老婆
getWife("大老婆","小老婆","俏老婆","刁蛮老婆","乖老婆","送上门老婆");
// 换一批老婆
getWife("超越韦小宝的老婆");
 结果:
合法老婆;大老婆;小老婆;俏老婆;刁蛮老婆;乖老婆;送上门老婆
合法老婆;超越韦小宝的老婆
实例2:
var curryWeight = function(fn) {
 var _fishWeight = [];
 return function() {
  if (arguments.length === 0) {
   return fn.apply(null, _fishWeight);
  } else {
   _fishWeight = _fishWeight.concat([].slice.call(arguments));
  }
 }
};
var fishWeight = 0;
var addWeight = curryWeight(function() {
 var i=0; len = arguments.length;
 for (i; i<len; i+=1) {
  fishWeight += arguments[i];
 }
});
addWeight(2.3);
addWeight(6.5);
addWeight(1.2);
addWeight(2.5);
addWeight(); // 这里才计算
console.log(fishWeight); // 12.5

总结

以上所述是小编给大家介绍的JS中的函数柯里化(currying),希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

Javascript 相关文章推荐
baidu博客的编辑友情链接的新的层窗口!经典~支持【FF】
Feb 09 Javascript
jQuery 相关控件的事件操作分解
Aug 03 Javascript
Javascript学习笔记7 原型链的原理
Jan 11 Javascript
JavaScript italics方法入门实例(把字符串显示为斜体)
Oct 17 Javascript
JS中三目运算符和if else的区别分析与示例
Nov 21 Javascript
JavaScript中标识符提升问题
Jun 11 Javascript
实例解析jQuery插件EasyUI最常用的表单验证规则
Nov 29 Javascript
高效的jquery数字滚动特效
Dec 17 Javascript
前端性能优化及技巧
May 06 Javascript
常用的javascript设计模式
Jan 11 Javascript
jQuery图片缩放插件smartZoom使用实例详解
Aug 25 jQuery
AngularJS 事件发布机制
Aug 28 Javascript
JavaScript实现简单评论功能
Aug 17 #Javascript
vue绑定class与行间样式style详解
Aug 16 #Javascript
Vue的MVVM实现方法
Aug 16 #Javascript
利用JS制作万年历的方法
Aug 16 #Javascript
原生JavaScript来实现对dom元素class的操作方法(推荐)
Aug 16 #Javascript
React Native 集成jpush-react-native的示例代码
Aug 16 #Javascript
jQuery实现全选、反选和不选功能
Aug 16 #jQuery
You might like
异世界新番又来了,同样是从零开始,男主的年龄降到5岁
2020/04/09 日漫
php性能优化分析工具XDebug 大型网站调试工具
2011/05/22 PHP
PHP将页面中点击数量高的链接进行高亮显示的方法
2016/05/30 PHP
jQuery1.5.1 animate方法源码阅读
2011/04/05 Javascript
javascript开发技术大全-第1章javascript概述
2011/07/03 Javascript
jquery focus(fn),blur(fn)方法实例代码
2011/12/16 Javascript
简易的投票系统以及js刷票思路和方法
2015/04/07 Javascript
深入解析JavaScript中的立即执行函数
2016/05/21 Javascript
Angular 4依赖注入学习教程之Injectable装饰器(六)
2017/06/04 Javascript
ionic2自定义cordova插件开发以及使用(Android)
2017/06/19 Javascript
JS原生数据双向绑定实现代码
2017/08/14 Javascript
微信小程序使用input组件实现密码框功能【附源码下载】
2017/12/11 Javascript
微信小程序scroll-view组件实现滚动动画
2018/01/31 Javascript
Bootstrap开发中Tab标签页切换图表显示问题的解决方法
2018/07/13 Javascript
web页面和微信小程序页面实现瀑布流效果
2018/09/26 Javascript
详解Vue-axios 设置请求头问题
2018/12/06 Javascript
vue中通过使用$attrs实现组件之间的数据传递功能
2019/09/01 Javascript
Vue 打包的静态文件不能直接运行的原因及解决办法
2020/11/19 Vue.js
python使用PyFetion来发送短信的例子
2014/04/22 Python
利用python获取某年中每个月的第一天和最后一天
2016/12/15 Python
使用apidocJs快速生成在线文档的实例讲解
2018/02/07 Python
django中静态文件配置static的方法
2018/05/20 Python
python_opencv用线段画封闭矩形的实例
2018/12/05 Python
使用Python批量修改文件名的代码实例
2019/01/24 Python
Python集合基本概念与相关操作实例分析
2019/10/30 Python
Python几种常见算法汇总
2020/06/02 Python
将世界上最美丽的摄影作品转化为艺术作品:Photos.com
2017/11/28 全球购物
英国异国风情旅游网站:Travel Talk Tours(团体旅游、探险旅游、帆船假期)
2018/07/26 全球购物
String、StringBuffer、StringBuilder有区别
2015/09/18 面试题
优秀家长事迹材料
2014/05/17 职场文书
房产公证书格式
2015/01/26 职场文书
python 统计代码耗时的几种方法分享
2021/04/02 Python
SQLServer2008提示评估期已过解决方案
2021/04/12 SQL Server
解决pytorch 损失函数中输入输出不匹配的问题
2021/06/05 Python
利用 SQL Server 过滤索引提高查询语句的性能分析
2021/07/15 SQL Server
CSS中float高度塌陷问题的四种解决方案
2022/04/18 HTML / CSS