深入理解JavaScript系列(49):Function模式(上篇)

yibin 2015-02-09 建站源码 577

介绍


本篇主要是介绍Function方面使用的一些技巧(上篇),利用Function特性可以编写出很多非常有意思的代码,本篇主要包括:回调模式、配置对象、返回函数、分布程序、柯里化(Currying)。

回调函数


JavaScript中,当一个函数A作为另外一个函数B的其中一个参数时,则函数A称为回调函数,即A可以在函数B的周期内执行(开始、中间、结束时均可)。
举例来说,有一个函数用于生成node
双击代码全选
1 var complexComputation = function () { };


 

有一个findNodes函数声明用于查找所有的节点,然后通过callback回调进行执行代码。
双击代码全选
1

2

3

4

5

6

7

8

9

10

11

12

13
var findNodes = function (callback) {

var nodes = [];

    

var node = complexComputation();

    

// 如果回调函数可用,则执行它

if (typeof callback === "function") {

callback(node);

}

    

nodes.push(node);

return nodes;

};


 

关于callback的定义,我们可以事先定义好来用:
双击代码全选
1

2

3

4

5

6

7
// 定义callback

var hide = function (node) {

node.style.display = "none";

};

    

// 查找node,然后隐藏所有的node

var hiddenNodes = findNodes(hide);


 

也可以直接在调用的时候使用匿名定义,如下:
双击代码全选
1

2

3

4
// 使用匿名函数定义callback

var blockNodes = findNodes(function (node) {

node.style.display = 'block';

});


 

我们平时用的最多的,估计就数jQuery的ajax方法的调用了,通过在done/faild上定义callback,以便在ajax调用成功或者失败的时候做进一步处理,代码如下(本代码基于jquery1.8版):
双击代码全选
1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17
var menuId = $("ul.nav").first().attr("id");

var request = $.ajax({

  url: "script.php",

  type: "POST",

  data: {id : menuId},

  dataType: "html"

});

    

//调用成功时的回调处理

request.done(function(msg) {

  $("#log").html( msg );

});

    

//调用失败时的回调处理

request.fail(function(jqXHR, textStatus) {

  alert( "Request failed: " + textStatus );

});


 

配置对象


如果一个函数(或方法)的参数只有一个参数,并且参数为对象字面量,我们则称这种模式为配置对象模式。例如,如下代码:
双击代码全选
1

2

3

4

5

6
var conf = {

    username:"shichuan",

    first:"Chuan",

    last:"Shi"

};

addPerson(conf);


 

则在addPerson内部,就可以随意使用conf的值了,一般用于初始化工作,例如jquery里的ajaxSetup也就是这种方式来实现的:
双击代码全选
1

2

3

4

5

6

7

8

9

10
// 事先设置好初始值

$.ajaxSetup({

   url: "/xmlhttp/",

   global: false,

   type: "POST"

    

 });

    

// 然后再调用

 $.ajax({ data: myData });


 

另外,很多jquery的插件也有这种形式的传参,只不过也可以不传,不传的时候则就使用默认值了。

返回函数


返回函数,则是指在一个函数的返回值为另外一个函数,或者根据特定的条件灵活创建的新函数,示例代码如下:
双击代码全选
1

2

3

4

5

6

7

8

9

10

11

12
var setup = function () {

    console.log(1);

    return function () {

        console.log(2);

    };

};

    

// 调用setup 函数

var my = setup(); // 输出 1

my(); // 输出 2

// 或者直接调用也可

setup()();


 

或者你可以利用闭包的特性,在setup函数里记录一个私有的计数器数字,通过每次调用来增加计数器,代码如下:
双击代码全选
1

2

3

4

5

6

7

8

9

10

11

12
var setup = function () {

    var count = 0;

    return function () {

        return ++count;

    };

};

    

// 用法

var next = setup();

next(); // 返回 1

next(); // 返回 2

next(); // 返回 3


 

偏应用


这里的偏应用,其实是将参数的传入工作分开进行,在有的时候一系列的操作可能会有某一个或几个参数始终完全一样,那么我们就可以先定义一个偏函数,然后再去执行这个函数(执行时传入剩余的不同参数)。
举个例子,代码如下:
双击代码全选
1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29
var partialAny = (function (aps) {

    

    // 该函数是你们自执行函数表达式的结果,并且赋值给了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);


 

使用方式如下:
双击代码全选
1

2

3

4

5

6

7

8

9

10
// 定义处理函数

function hex(r, g, b) {

    return '#' + r + g + b;

}

    

//定义偏函数, 将hex的第一个参数r作为不变的参数值ff

var redMax = partialAny(hex, 'ff', partialAny._, partialAny._);

    

// 新函数redMax的调用方式如下,只需要传入2个参数了:

console.log(redMax('11', '22')); // "#ff1122"


 

如果觉得partialAny._太长,可以用__代替哦。
双击代码全选
1

2

3

4

5

6

7

8

9

10
var __ = partialAny._;

    

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', __, 'ff');

console.log(magentaMax('77'));


 

这样使用,就简洁多了吧。

Currying


Currying是函数式编程的一个特性,将多个参数的处理转化成单个参数的处理,类似链式调用。
举一个简单的add函数的例子:
双击代码全选
1

2

3

4

5

6

7

8

9
function add(x, y) {

    var oldx = x, oldy = y;

    if (typeof oldy === "undefined") { // partial

        return function (newy) {

            return oldx + newy;

        }

    }

    return x + y;

}


 

这样调用方式就可以有多种了,比如:

扫码添加微信

13013082126 扫描微信 建站咨询 优化咨询