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); //
addOne(3); //
这里实现有两个重要的知识点,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); //
addOne(3); //
由于函数可以作为参数,所以我们还可以实现函数绑定。
部分函数应用
部分函数应用可以解释为,将一个函数和它一些的参数绑定,然后返回一个新的函数,这个新函数继续接收剩下未绑定的参数。
它与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); //
addOne(3); //
下面分析一下是怎么工作的,为了将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。
javascript中的部分函数应用的更多相关文章
- 浅谈JavaScript中的函数问题
前面的话:JavaScript可运行在所有主要平台的主流浏览器上,也可运行在每一个主流操作系统的服务器端上.所以呢,要想成为一名优秀的全栈工程师,必须懂得JavaScript语言.这是我整理的JS的部 ...
- javascript中的Array对象 —— 数组的合并、转换、迭代、排序、堆栈
Array 是javascript中经常用到的数据类型.javascript 的数组其他语言中数组的最大的区别是其每个数组项都可以保存任何类型的数据.本文主要讨论javascript中数组的声明.转换 ...
- javascript中的this与函数讲解
前言 javascript中没有块级作用域(es6以前),javascript中作用域分为函数作用域和全局作用域.并且,大家可以认为全局作用域其实就是Window函数的函数作用域,我们编写的js代码, ...
- JavaScript 中的数据类型
Javascript中的数据类型有以下几种情况: 基本类型:string,number,boolean 特殊类型:undefined,null 引用类型:Object,Function,Date,Ar ...
- javascript中的操作符详解1
好久没有写点什么了,根据博主的技术,仍然写一点javascript新手入门文章,接下来我们一起来探讨javascript的操作符. 一.前言 javascript中有许多操作符,但是许多初学者并不理解 ...
- 掌握javascript中的最基础数据结构-----数组
这是一篇<数据结构与算法javascript描述>的读书笔记.主要梳理了关于数组的知识.部分内容及源码来自原作. 书中第一章介绍了如何配置javascript运行环境:javascript ...
- javascript中变量提升的理解
网上找了两个经典的例子 var foo = 1; function bar() { if (!foo) { var foo = 10; } alert(foo); } bar(); // 10 var ...
- 前端开发:面向对象与javascript中的面向对象实现(二)构造函数与原型
前端开发:面向对象与javascript中的面向对象实现(二)构造函数与原型 前言(题外话): 有人说拖延症是一个绝症,哎呀治不好了.先不说这是一个每个人都多多少少会有的,也不管它究竟对生活有多么大的 ...
- 简单分析JavaScript中的面向对象
初学JavaScript的时候有人会认为JavaScript不是一门面向对象的语言,因为JS是没有类的概念的,但是这并不代表JavaScript没有对象的存在,而且JavaScript也提供了其它的方 ...
随机推荐
- 将常见对象转换成json字符串
public class JsonUtil { public static String objectTojson(Object obj) { StringBuilder json = new Str ...
- #include <bitset>
1 none();测试是否有越位 2 reset();全部清零 3 set(7, 0);把第7个字符改成0,操作二进制位 4 to_string();转换为字符串 5 to_ulong();转换为无符 ...
- OC中如何把字典中的数据拼接成url字符串
在使用objective-c语言开发iOS应用中,会向服务器通过URL请求一些数据,因此对URL的拼接肯定少不了.而在iOS中,我们一般是通过将字典中的数据拼接成我们要请求的URL字符串,那这个是怎么 ...
- autolisp 列表 resbuf
有以下 list : (1 2 3 (4 5 6) "Properties" (("id" . 3) ("name" . "hel ...
- 动态的 css——less
less 是一种样式语言,它将 css 赋予了动态语言的特性,如变量. 继承. 运算. 函数.less 既可以在客户端上运行(支持 ie6+,webkit,firefox),也可以借助 Node.js ...
- RMAN备份之非归档模式下的备份
Backing Up a Database in NOARCHIVELOG Mode:1.Log into RMAN2.Shutdown immediate from RMAN3.Startup mo ...
- JS函数的四种调用模式
函数在js中具有四种身份,分别为函数.方法.构造函数.apply或call调用 函数调用 函数调用模式中this指全局对象(window) var f1 = function() { alert ...
- JavaScript阻止事件冒泡
今天在自学敲代码的时候发现了一个问题,当时的例子如下: <!DOCTYPE html> <html lang="en"> <head> < ...
- VMware linux 增加根目录空间 (使用图形分区工具gparted LiveCd)
写这篇文章的原因: 最近要给服务器Centos上的ruby版本升级,由于是第一次升级,不敢直接在服务器上操作. 所以在我的winxp上装了Vmware ,又在Vmware中装了Centos5.2. 用 ...
- JAVAscript——菜单下拉列表练习(阻止事件冒泡)
下拉列表框,鼠标点击文本框,出现下拉,鼠标(离开的时候或者点击网页其他位置时)下拉列表消失.鼠标放到下拉列表的某一项上变背景色,点击下拉列表的某一项将该项的值显示在文本框内,然后下拉列表消失. < ...