欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

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

 更新時間:2015年03月04日 10:59:32   投稿:junjie  
這篇文章主要介紹了深入理解JavaScript系列(49):Function模式(上篇),本文講解了回調函數(shù)、配置對象、返回函數(shù)、偏應用、Currying等內容,需要的朋友可以參考下

介紹

本篇主要是介紹Function方面使用的一些技巧(上篇),利用Function特性可以編寫出很多非常有意思的代碼,本篇主要包括:回調模式、配置對象、返回函數(shù)、分布程序、柯里化(Currying)。

回調函數(shù)

在JavaScript中,當一個函數(shù)A作為另外一個函數(shù)B的其中一個參數(shù)時,則函數(shù)A稱為回調函數(shù),即A可以在函數(shù)B的周期內執(zhí)行(開始、中間、結束時均可)。

舉例來說,有一個函數(shù)用于生成node

復制代碼 代碼如下:

var complexComputation = function () { /* 內部處理,并返回一個node*/};

有一個findNodes函數(shù)聲明用于查找所有的節(jié)點,然后通過callback回調進行執(zhí)行代碼。
復制代碼 代碼如下:

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

var node = complexComputation();

// 如果回調函數(shù)可用,則執(zhí)行它
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);

也可以直接在調用的時候使用匿名定義,如下:

復制代碼 代碼如下:

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

我們平時用的最多的,估計就數(shù)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 );
});

配置對象

如果一個函數(shù)(或方法)的參數(shù)只有一個參數(shù),并且參數(shù)為對象字面量,我們則稱這種模式為配置對象模式。例如,如下代碼:

復制代碼 代碼如下:

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

則在addPerson內部,就可以隨意使用conf的值了,一般用于初始化工作,例如jquery里的ajaxSetup也就是這種方式來實現(xiàn)的:
復制代碼 代碼如下:

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

 });

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

另外,很多jquery的插件也有這種形式的傳參,只不過也可以不傳,不傳的時候則就使用默認值了。

返回函數(shù)

返回函數(shù),則是指在一個函數(shù)的返回值為另外一個函數(shù),或者根據(jù)特定的條件靈活創(chuàng)建的新函數(shù),示例代碼如下:

復制代碼 代碼如下:

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

// 調用setup 函數(shù)
var my = setup(); // 輸出 1
my(); // 輸出 2
// 或者直接調用也可
setup()();


或者你可以利用閉包的特性,在setup函數(shù)里記錄一個私有的計數(shù)器數(shù)字,通過每次調用來增加計數(shù)器,代碼如下:
復制代碼 代碼如下:

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

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

偏應用

這里的偏應用,其實是將參數(shù)的傳入工作分開進行,在有的時候一系列的操作可能會有某一個或幾個參數(shù)始終完全一樣,那么我們就可以先定義一個偏函數(shù),然后再去執(zhí)行這個函數(shù)(執(zhí)行時傳入剩余的不同參數(shù))。

舉個例子,代碼如下:

復制代碼 代碼如下:

var partialAny = (function (aps) {

    // 該函數(shù)是你們自執(zhí)行函數(shù)表達式的結果,并且賦值給了partialAny變量
    function func(fn) {
        var argsOrig = aps.call(arguments, 1);
        return function () {
            var args = [],
                argsPartial = aps.call(arguments),
                i = 0;

            // 變量所有的原始參數(shù)集,
            // 如果參數(shù)是partialAny._ 占位符,則使用下一個函數(shù)參數(shù)對應的值
            // 否則使用原始參數(shù)里的值
            for (; i < argsOrig.length; i++) {
                args[i] = argsOrig[i] === func._
                            ? argsPartial.shift()
                            : argsOrig[i];
            }

            // 如果有任何多余的參數(shù),則添加到尾部
            return fn.apply(this, args.concat(argsPartial));
        };
    }

    // 用于占位符設置
    func._ = {};

    return func;
})(Array.prototype.slice);

使用方式如下:

復制代碼 代碼如下:

// 定義處理函數(shù)
function hex(r, g, b) {
    return '#' + r + g + b;
}

//定義偏函數(shù), 將hex的第一個參數(shù)r作為不變的參數(shù)值ff
var redMax = partialAny(hex, 'ff', partialAny._, partialAny._);

// 新函數(shù)redMax的調用方式如下,只需要傳入2個參數(shù)了:
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是函數(shù)式編程的一個特性,將多個參數(shù)的處理轉化成單個參數(shù)的處理,類似鏈式調用。

舉一個簡單的add函數(shù)的例子:

復制代碼 代碼如下:

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函數(shù):
復制代碼 代碼如下:

// 第一個參數(shù)為要應用的function,第二個參數(shù)是需要傳入的最少參數(shù)個數(shù)
function curry(func, minArgs) {
    if (minArgs == undefined) {
        minArgs = 1;
    }

    function funcWithArgsFrozen(frozenargs) {
        return function () {
            // 優(yōu)化處理,如果調用時沒有參數(shù),返回該函數(shù)本身
            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([]);
}

這樣,我們就可以隨意定義我們的業(yè)務行為了,比如定義加法:

復制代碼 代碼如下:

var plus = curry(function () {
    var result = 0;
    for (var i = 0; i < arguments.length; ++i) {
        result += arguments[i];
    }
    return result;
}, 2);

使用方式,真實多種多樣哇。
復制代碼 代碼如下:

plus(3, 2) // 正常調用
plus(3) // 偏應用,返回一個函數(shù)(返回值為3+參數(shù)值)
plus(3)(2) // 完整應用(返回5)
plus()(3)()()(2) // 返回 5
plus(3, 2, 4, 5) // 可以接收多個參數(shù)
plus(3)(2, 3, 5) // 同理

如下是減法的例子
復制代碼 代碼如下:

var minus = curry(function (x) {
    var result = x;
    for (var i = 1; i < arguments.length; ++i) {
        result -= arguments[i];
    }
    return result;
}, 2);

或者如果你想交換參數(shù)的順序,你可以這樣定義
復制代碼 代碼如下:

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

總結

JavaScript里的Function有很多特殊的功效,可以利用閉包以及arguments參數(shù)特性實現(xiàn)很多不同的技巧,下一篇我們將繼續(xù)介紹利用Function進行初始化的技巧。

相關文章

  • 關于uni-app頁面Page和組件Component生命周期執(zhí)行的先后順序

    關于uni-app頁面Page和組件Component生命周期執(zhí)行的先后順序

    這篇文章主要介紹了關于uni-app頁面Page和組件Component生命周期執(zhí)行的先后順序,文中提供了具體的代碼,還不清楚的朋友可以來學習一下
    2023-04-04
  • JavaScript初級教程(第五課續(xù))

    JavaScript初級教程(第五課續(xù))

    JavaScript初級教程(第五課續(xù))...
    2007-04-04
  • 關于javascript事件響應的基礎語法總結(必看篇)

    關于javascript事件響應的基礎語法總結(必看篇)

    下面小編就為大家?guī)硪黄P于javascript事件響應的基礎語法總結(必看篇)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧,祝大家游戲愉快哦
    2016-12-12
  • javascript的函數(shù)作用域

    javascript的函數(shù)作用域

    javascript中沒有塊級作用域。取而代之的javascript使用的是函數(shù)作用域(function scope):變量在聲明它的函數(shù)體以及這個函數(shù)體嵌套的任意函數(shù)體內都是有定義的。
    2014-11-11
  • ES6基礎語法之Map和Set對象

    ES6基礎語法之Map和Set對象

    這篇文章介紹了ES6基礎語法之Map和Set對象,文中通過示例代碼介紹的非常詳細。對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-05-05
  • 簡介JavaScript中的getSeconds()方法的使用

    簡介JavaScript中的getSeconds()方法的使用

    這篇文章主要介紹了簡介JavaScript中的getSeconds()方法的使用,是JS入門學習中的基礎知識,需要的朋友可以參考下
    2015-06-06
  • fetch()函數(shù)說明與使用方法詳解

    fetch()函數(shù)說明與使用方法詳解

    fetch()是XMLHttpRequest的升級版,用于在JavaScript腳本里面發(fā)出 HTTP請求,本文章向大家介紹fetch()的用法,主要包括fetch()的用法使用實例、應用技巧、基本知識點總結和需要注意事項,具有一定的參考價值,需要的朋友可以參考一下。
    2022-11-11
  • javascript的 {} 語句塊詳解

    javascript的 {} 語句塊詳解

    這篇文章主要介紹了javascript的 {} 語句塊詳解的相關資料,需要的朋友可以參考下
    2016-02-02
  • JavaScript基本對象

    JavaScript基本對象

    JavaScript基本對象...
    2007-01-01
  • JS Date函數(shù)整理方便使用

    JS Date函數(shù)整理方便使用

    JS Date 對象用于處理日期和時間,在項目中使用很廣泛,下面是本文對JS Date函數(shù)的整理,感興趣的朋友可以參考下
    2013-10-10

最新評論