javascript中的部分函数应用
这篇文章写的很全面,不过也啰嗦:http://benalman.com/news/2012/09/partial-application-in-javascript/
这篇文章是神级运用:http://osteele.com/sources/javascript/functional/
绑定变量
假设我们函数的部分参数已经固定,我们可以绑定这个参数,生成新的函数。
// 一般函数 function add(a, b) { return a + b; } // 特定情况的函数生成函数 function makeAdder(a) { return function(b) { return a + b; }; } // 特定函数 var addOne = makeAdder(1); addOne(2); // 3 addOne(3); // 4
这里实现有两个重要的知识点,1.javascript允许函数访问外部变量(详细见javascript的闭包概念)。2.在javascript里,函数是可以将函数作为参数和返回值。
绑定函数
有时候我们需要绑定的不只是数值
// 特定函数生成函数 function bindFirstArg(fn, a) { return function(b) { return fn(a, b); }; } // 一般函数 function add(a, b) { return a + b; } // 特定函数 var addOne = bindFirstArg(add, 1); addOne(2); // 3 addOne(3); // 4
由于函数可以作为参数,所以我们还可以实现函数绑定。
部分函数应用
部分函数应用可以解释为,将一个函数和它一些的参数绑定,然后返回一个新的函数,这个新函数继续接收剩下未绑定的参数。
它与bind()方法原理上有些相似。
注意:arguments 对象是一个类数组对象,当函数被调用时创建,只有在函数内部可以引用,它包括所有传入函数的参数。
下面的部分应用函数,和使用实例。
function partial(fn /*, args...*/) { var slice = Array.prototype.slice; var args = slice.call(arguments, 1); return function() { return fn.apply(this, args.concat(slice.call(arguments, 0))); }; }
下面是一个使用部分函数的例子:
function add(a, b) { return a + b; } var addOne = partial(add, 1); addOne(2); // 3 addOne(3); // 4
下面分析一下是怎么工作的,为了将arguments转化为数组,就要使用数组的方法Array.prototype.slice,因为要使用两次,所以存储到slice缓存。partial函数将它的参数除第一个fn外,存储到args上,以便返回函数访问。当返回函数f调用时,再将新arguments对象转化为数组,利用数组方法Array.prototype.concat将两个数组合并,再利用apply调用fn,this是window,参数就是先前合并的参数,f的返回值就是fn的返回值。
注意,他是利用apply可以接收数组的特性,将两个函数的参数转成数组再合并,以此保证绑定函数参数的长度可以不固定。
也可以这样调用函数
Function.prototype.partial = function() { var fn = this, args = Array.prototype.slice.call(arguments); return function() { return fn.apply(this, args.concat( Array.prototype.slice.call(arguments))); }; };
javascript中函数this指向调用它的对象。又函数的都是继承自函数的原型,所以可以String.prototype.split.partial()调用partial,而this指向的是split,这也就是partial的fn。
如何跳跃式绑定参数值?
Function.prototype.partial = function(){ var fn = this, args = Array.prototype.slice.call(arguments); return function(){ var arg = 0; for ( var i = 0; i < args.length && arg < arguments.length; i++ ) if ( args[i] === undefined ) args[i] = arguments[arg++]; return fn.apply(this, args); }; };
思路就是,如果想跳跃的参数值为undefined,在返回函数f里,对先前值为undefined进行覆盖。
部分函数应用到底有什么用?
以下例子基于上面的partial。
String.prototype.csv = String.prototype.split.partial(/,\s*/); var results = "John, Resig, Boston".csv(); alert( (results[1] == "Resig") + " The text values were split properly" );
这里我们返回一个函数,保存到String.prototype.csv中,每个字符串默认就可以用csv进行字符串拆分。
var delay = setTimeout.partial(undefined, 10); delay(function(){ alert( "A call to this function will be temporarily delayed." ); });
这里我们返回一个默认延迟为10的函数,可以一定程度上简化调用。
var bindClick = document.body.addEventListener .partial("click", undefined, false); bindClick(function(){ alert( "Click event bound via curried function." ); });
简化事件监听。
这个技术可以用于特定情况,构造简化的API。