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

11,816次阅读
没有评论

共计 3861 个字符,预计需要花费 10 分钟才能阅读完成。

介绍

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

回调函数

在 JavaScript 中,当一个函数 A 作为另外一个函数 B 的其中一个参数时,则函数 A 称为回调函数,即 A 可以在函数 B 的周期内执行(开始、中间、结束时均可)。

举例来说,有一个函数用于生成 node

var complexComputation = function () { /* 内部处理,并返回一个 node*/};

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

var findNodes = function (callback) {var nodes = [];

    var node = complexComputation();

    // 如果回调函数可用,则执行它
    if (typeof callback === "function") {callback(node);
    }

    nodes.push(node);
    return nodes;
};

关于 callback 的定义,我们可以事先定义好来用:

// 定义 callback
var hide = function (node) {node.style.display = "none";};

// 查找 node,然后隐藏所有的 node
var hiddenNodes = findNodes(hide);

也可以直接在调用的时候使用匿名定义,如下:

// 使用匿名函数定义 callback
var blockNodes = findNodes(function (node) {node.style.display = 'block';});

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

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);
});

配置对象

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

var conf = {
    username:"shichuan",
    first:"Chuan",
    last:"Shi"
};
addPerson(conf);

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

// 事先设置好初始值
$.ajaxSetup({
   url: "/xmlhttp/",
   global: false,
   type: "POST"

 });

// 然后再调用
 $.ajax({data: myData});

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

返回函数

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

var setup = function () {console.log(1);
    return function () {console.log(2);
    };
};

// 调用 setup 函数
var my = setup(); // 输出 1
my(); // 输出 2
// 或者直接调用也可
setup()();

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

var setup = function () {
    var count = 0;
    return function () {return ++count;};
};

// 用法
var next = setup();
next(); // 返回 1
next(); // 返回 2
next(); // 返回 3

偏应用

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

举个例子,代码如下:

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 

使用方式如下:

// 定义处理函数
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._太长,可以用__代替哦。

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 函数的例子:

function add(x, y) {
    var oldx = x, oldy = y;
    if (typeof oldy === "undefined") { // partial
        return function (newy) {return oldx + newy;}
    }
    return x + y;
}

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

// 测试
typeof add(5); // "function"
add(3)(4); // 7

// 也可以这样调用
var add2000 = add(2000);
add2000(10); // 2010

接下来,我们来定义一个比较通用的 currying 函数:

// 第一个参数为要应用的 function,第二个参数是需要传入的最少参数个数
function curry(func, minArgs) {if (minArgs == undefined) {minArgs = 1;}

    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([]);
}

这样,我们就可以随意定义我们的业务行为了,比如定义加法:

var plus = curry(function () {
    var result = 0;
    for (var i = 0; i 

使用方式,真实多种多样哇。

plus(3, 2) // 正常调用
plus(3) // 偏应用,返回一个函数(返回值为 3 + 参数值)plus(3)(2) // 完整应用(返回 5)plus()(3)()()(2) // 返回 5
plus(3, 2, 4, 5) // 可以接收多个参数
plus(3)(2, 3, 5) // 同理

如下是减法的例子

var minus = curry(function (x) {
    var result = x;
    for (var i = 1; i 

或者如果你想交换参数的顺序,你可以这样定义

var flip = curry(function (func) {return curry(function (a, b) {return func(b, a);
    }, 2);
});

总结

JavaScript 里的 Function 有很多特殊的功效,可以利用闭包以及 arguments 参数特性实现很多不同的技巧,下一篇我们将继续介绍利用 Function 进行初始化的技巧。

    正文完
     0
    Yojack
    版权声明:本篇文章由 Yojack 于2024-09-24发表,共计3861字。
    转载说明:
    1 本网站名称:优杰开发笔记
    2 本站永久网址:https://yojack.cn
    3 本网站的文章部分内容可能来源于网络,仅供大家学习与参考,如有侵权,请联系站长进行删除处理。
    4 本站一切资源不代表本站立场,并不代表本站赞同其观点和对其真实性负责。
    5 本站所有内容均可转载及分享, 但请注明出处
    6 我们始终尊重原创作者的版权,所有文章在发布时,均尽可能注明出处与作者。
    7 站长邮箱:laylwenl@gmail.com
    评论(没有评论)