介绍
本篇主要是介绍Function方面使用的一些技巧(上篇),利用Function特性可以编写出很多非常有意思的代码,本篇主要包括:回调模式、配置对象、返回函数、分布程序、柯里化(Currying)。
回调函数
在JavaScript中,当一个函数A作为另外一个函数B的其中一个参数时,则函数A称为回调函数,即A可以在函数B的周期内执行(开始、中间、结束时均可)。
var node = complexComputation();
// 如果回调函数可用,则执行它 if (typeof callback === "function") { callback(node); }
nodes.push(node); return nodes; };
关于callback的定义,我们可以事先定义好来用:
// 查找node,然后隐藏所有的node var hiddenNodes = findNodes(hide);
也可以直接在调用的时候使用匿名定义,如下:
//调用成功时的回调处理 request.done(function(msg) { $("#log").html( msg ); });
//调用失败时的回调处理 request.fail(function(jqXHR,textStatus) { alert( "Request Failed: " + textStatus ); });
配置对象
如果一个函数(或方法)的参数只有一个参数,并且参数为对象字面量,我们则称这种模式为配置对象模式。例如,如下代码:
});
// 然后再调用 $.ajax({ data: myData });
另外,很多jquery的插件也有这种形式的传参,只不过也可以不传,不传的时候则就使用默认值了。
返回函数
返回函数,则是指在一个函数的返回值为另外一个函数,或者根据特定的条件灵活创建的新函数,示例代码如下:
// 用法 var next = setup(); next(); // 返回 1 next(); // 返回 2 next(); // 返回 3
偏应用
这里的偏应用,其实是将参数的传入工作分开进行,在有的时候一系列的操作可能会有某一个或几个参数始终完全一样,那么我们就可以先定义一个偏函数,然后再去执行这个函数(执行时传入剩余的不同参数)。
举个例子,代码如下:
// 该函数是你们自执行函数表达式的结果,并且赋值给了partialAny变量 function func(fn) { var argsOrig = aps.call(arguments,1); return function () { var args = [], argsPartial = aps.call(arguments), i = 0;
// 变量所有的原始参数集, // 如果参数是partialAny._ 占位符,则使用下一个函数参数对应的值 // 否则使用原始参数里的值 for (; i < argsOrig.length; i++) { args[i] = argsOrig[i] === func._ ? argsPartial.shift() : argsOrig[i]; }
// 如果有任何多余的参数,则添加到尾部 return fn.apply(this,args.concat(argsPartial)); }; }
// 用于占位符设置 func._ = {};
return func; })(Array.prototype.slice);
使用方式如下:
//定义偏函数,将hex的第一个参数r作为不变的参数值ff var redMax = partialAny(hex,'ff',partialAny._,partialAny._);
// 新函数redMax的调用方式如下,只需要传入2个参数了: console.log(redMax('11','22')); // "#ff1122"
var greenMax = partialAny(hex,__,'ff'); console.log(greenMax('33','44'));
var blueMax = partialAny(hex,'ff'); console.log(blueMax('55','66'));
var magentaMax = partialAny(hex,'ff'); console.log(magentaMax('77'));
Currying
Currying是函数式编程的一个特性,将多个参数的处理转化成单个参数的处理,类似链式调用。
举一个简单的add函数的例子:
// 也可以这样调用 var add2000 = add(2000); add2000(10); // 2010
function funcWithArgsFrozen(frozenargs) { return function () { // 优化处理,如果调用时没有参数,返回该函数本身 var args = Array.prototype.slice.call(arguments); var newArgs = frozenargs.concat(args); if (newArgs.length >= minArgs) { return func.apply(this,newArgs); } else { return funcWithArgsFrozen(newArgs); } }; }
return funcWithArgsFrozen([]); }
这样,我们就可以随意定义我们的业务行为了,比如定义加法: