通过实例了解Javascript柯里化流程
函数式编程是一种如今比较流行的编程范式,它主张将函数作为参数进行传递,然后返回一个没有副作用的函数,说白了,就是希望一个函数只做一件事情。
像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
如上实现就很完善,该工具函数的实现起来就一句话
利用闭包将函数的参数储存起来,等参数达到一定数量时执行函数。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持狼蚁SEO。
编程语言
- 如何快速学会编程 如何快速学会ug编程
- 免费学编程的app 推荐12个免费学编程的好网站
- 电脑怎么编程:电脑怎么编程网咯游戏菜单图标
- 如何写代码新手教学 如何写代码新手教学手机
- 基础编程入门教程视频 基础编程入门教程视频华
- 编程演示:编程演示浦丰投针过程
- 乐高编程加盟 乐高积木编程加盟
- 跟我学plc编程 plc编程自学入门视频教程
- ug编程成航林总 ug编程实战视频
- 孩子学编程的好处和坏处
- 初学者学编程该从哪里开始 新手学编程从哪里入
- 慢走丝编程 慢走丝编程难学吗
- 国内十强少儿编程机构 中国少儿编程机构十强有
- 成人计算机速成培训班 成人计算机速成培训班办
- 孩子学编程网上课程哪家好 儿童学编程比较好的
- 代码编程教学入门软件 代码编程教程