Javascript柯里化实现原理及作用解析


Posted in Javascript onOctober 22, 2020

函数式编程是一种如今比较流行的编程范式,它主张将函数作为参数进行传递,然后返回一个没有副作用的函数,说白了,就是希望一个函数只做一件事情。

像JavaScript,Haskell,Clojure等编程语言都支持函数式编程。

这种编程思想涵盖了三个重要的概念:

  • 纯函数
  • 柯里化
  • 高阶函数

而这篇文章主要是想向大家讲清楚柯里化这个概念。

什么是柯里化

首先我们先来看一个例子:

function sum(a, b, c) {
return a + b + c;
}
// 调用
sum(1, 2, 3); // 6

上述函数实现的是将a,b,c三个参数相加,改写为柯里化函数如下:

function sum(a) {
return function (b) {
return function(c) {
return a + b + c;
}
}
}
// 调用
let sum1 = sum(1);
let sum2 = sum1(2);
sum2(3); // 6

所谓柯里化就是把具有较多参数的函数转换成具有较少参数的函数的过程。

我们来一步步看上面那个柯里化函数做了什么,首先第一步调用了sum(1),此时变量sum1相当于:

sum1 = function(b) {
return function(c) {
// 注意此时变量a存在于闭包中,可以调用,a = 1
return a + b + c;
}
}

然后调用sum1(2),此时赋值给变量sum2相当于:

sum2 = function(c) {
// 变量a,b皆在闭包中, a = 1, b = 2
return a + b + c;
}

最后调用sum2(3),返回1 + 2 + 3的结果6;

这就是一个最简单的柯里化函数,是不是很简单呢?

柯里化函数的作用

那么问题来了,上面改写后的柯里化函数和原函数比起来代码多了不少,而且也不如原函数好理解,柯里化函数到底有什么用呢?

确实,柯里化函数在这里看起来的确是很臃肿,不实用,但在很多场景下他的作用是很大的,甚至很多人在不经意间已经在使用柯里化函数了。举一个简单的例子:

假设我们有一批的长方体,我们需要计算这些长方体的体积,实现一个如下函数:

function volume(length, width, height) {
  return length * width * height;
}
volume(200, 100, 200);
volume(200, 150, 100);
volume(200, 50, 80);
volume(100, 50, 60);

如上计算长方体的体积函数会发现存在很多相同长度的长方体,我们再用柯里化函数实现一下:

function volume(length, width, height) {
  return function(width) {
    return function(height) {
     return length * width * height;
      }
  }
}
let len200 = volume(200);
len200(100)(200);
len200(150)(100);
len200(50)(80);
volume(100)(50)(60);

如上,通过实现一个len200函数我们统一处理长度为200的长方体的体积,这就实现了参数复用。

我们再举一个只执行一次函数的例子:

function execOnce(fun) {
   let flag = true;
 return function() {
  if (flag) {
   fun && fun();
   flag = false;
  }
 }
}
let onceConsole = execOnce(function() {
  console.log('只打印一次');
});
onceConsole();
onceConsole();

如上,我们实现了一个execOnce函数,该函数接受一个函数参数,然后返回一个函数,变量flag存在闭包中,用来判断返回的函数是否执行过,onceConsole相当于:

let onceConsole = function() {
 if (flag) {
     (function() {
    console.log('只打印一次');
   })()
   flag = false;
  }
}

这也是柯里化函数的一个简单应用。

通用柯里化函数的实现

既然柯里化函数这么实用,那么我们能不能实现一个通用的柯里化函数呢?所谓通用,就是说该函数可以把函数参数转换为柯里化函数,看下第一版实现的代码:

// 第一版
var curry = function (fn) {
 var args = [].slice.call(arguments, 1);
 return function() {
  var newArgs = args.concat([].slice.call(arguments));
  return fn.apply(null, newArgs);
 };
};
 function add(a, b) {
  return a + b;
 }

var addFun = curry(add, 1, 2);
addFun() // 3
//或者
var addOne = curry(add, 1);

如上代码,我们接受一个函数作为参数,然后收集其它的参数,将这些参数传给这个函数参数去执行。但上面的代码有个问题,参数不够自由,比如我们想这么调用就会报错:

var addFun = curry(function(a, b,c) {
return a + b + c;
}, 1);
addFun(2)(3); // 报错 addFun(...) is not a function

这好像违背了我们参数复用的原则,改进如下:

function curry(fn, args) {
 var length = fn.length;
 args = args || [];
 return function(...rest) {
  var _args = [...args, ...rest];
  return _args.length < length
   ? curry.call(this, fn, _args)
  : fn.apply(this, _args);
 }
}
var fn = curry(function(a, b, c) {
 console.log(a + b + c);
});
fn('a', 'b', 'c'); // abc
fn('a', 'b')('c'); // abc
fn('a')('b')('c'); // abc

如上实现就很完善,该工具函数的实现总结起来就一句话:

利用闭包将函数的参数储存起来,等参数达到一定数量时执行函数。

后记

柯里化是以闭包为基础的,不理解闭包可能对柯里化的理解有所阻碍,希望通过这篇文章能让各位了解和理解JavaScript的柯里化。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
jquery 最简单的属性菜单
Oct 08 Javascript
判断用户的在线状态 onbeforeunload事件
Mar 05 Javascript
nodeType属性返回被选节点的节点类型介绍
Nov 22 Javascript
button没写type=button会导致点击时提交
Mar 06 Javascript
js换图片效果可进行定时操作
Jun 09 Javascript
字段太多jquey快速清空表单内容方法
Aug 21 Javascript
不到200行 JavaScript 代码实现富文本编辑器的方法
Jan 03 Javascript
jQuery实现判断上传图片类型和大小的方法示例
Apr 11 jQuery
详解webpack-dev-server 设置反向代理解决跨域问题
Apr 18 Javascript
vue集成kindeditor富文本的实现示例代码
Jun 07 Javascript
解决echarts vue数据更新,视图不更新问题(echarts嵌在vue弹框中)
Jul 20 Javascript
javascript实现贪吃蛇游戏(娱乐版)
Aug 17 Javascript
Node.js fs模块原理及常见用途
Oct 22 #Javascript
使用vue构建多页面应用的示例
Oct 22 #Javascript
vue 单页应用和多页应用的优劣
Oct 22 #Javascript
Javascript Symbol原理及使用方法解析
Oct 22 #Javascript
多个Vue项目部署到服务器的步骤记录
Oct 22 #Javascript
针对Vue路由history模式下Nginx后台配置操作
Oct 22 #Javascript
微信小程序基于高德地图API实现天气组件(动态效果)
Oct 22 #Javascript
You might like
PHP 缓存实现代码及详细注释
2010/05/16 PHP
在smarty模板中使用PHP函数的方法
2011/04/23 PHP
PHP超级全局变量数组小结
2012/10/04 PHP
在Nginx上部署ThinkPHP项目教程
2015/02/02 PHP
PHP下 Mongodb 连接远程数据库的实例代码
2017/08/30 PHP
阿里云Win2016安装Apache和PHP环境图文教程
2018/03/11 PHP
javascript 操作cookies及正确使用cookies的属性
2009/10/15 Javascript
锋利的jQuery 要点归纳(三) jQuery中的事件和动画(上:事件篇)
2010/03/24 Javascript
该如何加载google-analytics(或其他第三方)的JS
2010/05/13 Javascript
js获取网页高度(详细整理)
2012/12/28 Javascript
javascript 实现简单的table排序及table操作练习
2012/12/28 Javascript
A标签触发onclick事件而不跳转的多种解决方法
2013/06/27 Javascript
jquery.validate.js插件使用经验记录
2014/07/02 Javascript
为什么JS中eval处理JSON数据要加括号
2015/04/13 Javascript
JS实现把鼠标放到链接上出现滚动文字的方法
2016/04/06 Javascript
结合代码图文讲解JavaScript中的作用域与作用域链
2016/07/05 Javascript
jQuery基于BootStrap样式实现无限极地区联动
2016/08/26 Javascript
Vue数据驱动模拟实现1
2017/01/11 Javascript
easyui-datagrid特殊字符不能显示的处理方法
2017/04/12 Javascript
mui 打开新窗口的方式总结及注意事项
2017/08/20 Javascript
浅谈node中的cluster集群
2018/06/02 Javascript
[01:02:48]2018DOTA2亚洲邀请赛小组赛 A组加赛 Newbee vs Liquid
2018/04/03 DOTA
Python单例模式实例详解
2017/03/01 Python
Python生成器的使用方法和示例代码
2019/03/04 Python
用django-allauth实现第三方登录的示例代码
2019/06/24 Python
python中有关时间日期格式转换问题
2019/12/25 Python
python通用数据库操作工具 pydbclib的使用简介
2020/12/21 Python
python 制作磁力搜索工具
2021/03/04 Python
GAZMAN官网:澳大利亚领先的男装品牌
2019/12/19 全球购物
建筑专业自荐信
2013/10/18 职场文书
2014教师研修学习体会
2014/07/08 职场文书
2014矛盾纠纷排查调处工作总结
2014/12/09 职场文书
答谢词范文
2015/01/05 职场文书
CSS 新特性 contain控制页面的重绘与重排问题
2021/04/30 HTML / CSS
Python中的变量与常量
2021/11/11 Python
mysql全面解析json/数组
2022/07/07 MySQL