国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

underscore 系列之內(nèi)部函數(shù) restArgs

zzzmh / 2502人閱讀

摘要:與最后,使用我們的寫的函數(shù)重寫下函數(shù)系列系列目錄地址。系列預(yù)計寫八篇左右,重點介紹中的代碼架構(gòu)鏈?zhǔn)秸{(diào)用內(nèi)部函數(shù)模板引擎等內(nèi)容,旨在幫助大家閱讀源碼,以及寫出自己的。如果有錯誤或者不嚴(yán)謹(jǐn)?shù)牡胤剑垊?wù)必給予指正,十分感謝。

partial

在《 JavaScript 專題之偏函數(shù)》中,我們寫了一個 partial 函數(shù),用來固定函數(shù)的部分參數(shù),實現(xiàn)代碼如下:

// 這是文章中的第一版
function partial(fn) {
    var args = [].slice.call(arguments, 1);
    return function() {
        var newArgs = args.concat([].slice.call(arguments));
        return fn.apply(this, newArgs);
    };
};
rest parameter

ES6 為我們提供了剩余參數(shù)(rest parameter)語法,允許我們將一個不定數(shù)量的參數(shù)表示為一個數(shù)組。

function fn(a, b, ...args) {
   console.log(args); // [3, 4, 5]
}

fn(1, 2, 3, 4, 5)

我們可以利用這一特性簡化 partial 實現(xiàn)的代碼:

function partial(fn, ...args) {
    return function(...partialArgs) {
        var newArgs = args.concat(partialArgs);
        return fn.apply(this, newArgs);
    };
};

寫個 demo,測試一下:

function add(a, b) {
    return a + b;
}

var addOne = partial(add, 1);

console.log(addOne(2)); // 3
restArgs

如果不使用 ... 拓展操作符,僅用 ES5 的內(nèi)容,該怎么實現(xiàn)呢?

我們可以寫一個 restArgs 函數(shù),傳入一個函數(shù),使用函數(shù)的最后一個參數(shù)儲存剩下的函數(shù)參數(shù),使用效果如下:

var func = restArgs(function(a, b, c){
    console.log(c); // [3, 4, 5]
})

func(1, 2, 3, 4, 5)

我們來寫一版:

// 第一版
function restArgs(func) {
    return function(){
        // startIndex 表示使用哪個位置的參數(shù)用于儲存剩余的參數(shù)
        var startIndex = func.length - 1;
        var length = arguments.length - startIndex;

        var rest = Array(length)
        var index = 0;

        // 使用一個數(shù)組儲存剩余的參數(shù)
        // 以上面的例子為例,結(jié)果為:
        // rest [3, 4, 5]
        for (; index < length; index++) {
            rest[index] = arguments[index + startIndex]
        }

        // args [1, 2, undefined]
        var args = Array(startIndex + 1);
        for (index = 0; index < startIndex; index++) {
            args[index] = arguments[index]
        }

        // args [1, 2, [3, 4, 5]]
        args[startIndex] = rest;

        return func.apply(this, args)
    }
}
優(yōu)化

我們默認使用傳入的函數(shù)的最后一個參數(shù)儲存剩余的參數(shù),為了更加靈活,我們可以再增加一個參數(shù),用來指定 startIndex,如果沒有指定,就默認使用最后一個參數(shù)。

此外,注意,我們使用 Array(length) 創(chuàng)建數(shù)組,而 length 的計算方式是 arguments.length - startIndex,這個值有可能是負數(shù)!比如:

var func = restArgs(function(a, b, c, d){
    console.log(c) // 報錯
})

func(1, 2)

所以我們再寫一版:

// 第二版
function restArgs(func, startIndex) {
    startIndex = startIndex == null ? func.length - 1 : +startIndex;
    return function(){
        var length = Math.max(arguments.length - startIndex, 0);
        var rest = Array(length)
        var index = 0;
        for (; index < length; index++) {
            rest[index] = arguments[index + startIndex]
        }

        var args = Array(startIndex + 1);
        for (index = 0; index < startIndex; index++) {
            args[index] = arguments[index]
        }

        args[startIndex] = rest;
        return func.apply(this, args)
    }
}
性能優(yōu)化

如果是正常寫業(yè)務(wù),可能寫到這里就結(jié)束了,然而 underscore 考慮的更多,鑒于 call 的性能要高于 apply,所以 underscore 做了一個優(yōu)化:

// 第三版
var restArgs = function(func, startIndex) {
    startIndex = startIndex == null ? func.length - 1 : +startIndex;
    return function() {
        var length = Math.max(arguments.length - startIndex, 0),
            rest = Array(length),
            index = 0;

        for (; index < length; index++) {
            rest[index] = arguments[index + startIndex];
        }

        // 增加的部分
        switch (startIndex) {
            case 0:
                return func.call(this, rest);
            case 1:
                return func.call(this, arguments[0], rest);
            case 2:
                return func.call(this, arguments[0], arguments[1], rest);
        }

        var args = Array(startIndex + 1);
        for (index = 0; index < startIndex; index++) {
            args[index] = arguments[index];
        }

        args[startIndex] = rest;
        return func.apply(this, args);
    };
};

至此,restArgs 函數(shù)就完成了,underscore 很多函數(shù)比如 invoke、without、union、difference、bind、partial、bindAll、delay 都用到了 restArgs 函數(shù)。

當(dāng)使用 underscore 的時候,我們可以以 _.restArgs 的形式調(diào)用該函數(shù)。

restArgs 與 partial

最后,使用我們的寫的 restArgs 函數(shù)重寫下 partial 函數(shù):

var partial = restArgs(function(fn, args){
    return restArgs(function(partialArgs) {
        var newArgs = args.concat(partialArgs);
        return fn.apply(this, newArgs);
    })
})

function add(a, b, c) {
    return a + b + c;
}

var addOne = partial(add, 1);
console.log(addOne(2, 3)); // 6
underscore 系列

underscore 系列目錄地址:https://github.com/mqyqingfeng/Blog。

underscore 系列預(yù)計寫八篇左右,重點介紹 underscore 中的代碼架構(gòu)、鏈?zhǔn)秸{(diào)用、內(nèi)部函數(shù)、模板引擎等內(nèi)容,旨在幫助大家閱讀源碼,以及寫出自己的 undercore。

如果有錯誤或者不嚴(yán)謹(jǐn)?shù)牡胤剑垊?wù)必給予指正,十分感謝。如果喜歡或者有所啟發(fā),歡迎 star,對作者也是一種鼓勵。

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/90212.html

相關(guān)文章

  • underscore 的源碼該如何閱讀?

    摘要:所以它與其他系列的文章并不沖突,完全可以在閱讀完這個系列后,再跟著其他系列的文章接著學(xué)習(xí)。如何閱讀我在寫系列的時候,被問的最多的問題就是該怎么閱讀源碼我想簡單聊一下自己的思路。感謝大家的閱讀和支持,我是冴羽,下個系列再見啦 前言 別名:《underscore 系列 8 篇正式完結(jié)!》 介紹 underscore 系列是我寫的第三個系列,前兩個系列分別是 JavaScript 深入系列、...

    weknow619 評論0 收藏0
  • 窺探Underscore源碼系列-開篇

    摘要:他指示了一個對象的屬性,返回的將用來獲得該屬性對應(yīng)的值在上面的分析中,我們知道,當(dāng)傳入的是一個函數(shù)時,還要經(jīng)過一個叫的內(nèi)置函數(shù)才能獲得最終的所以此處的必然是優(yōu)化回調(diào)的作用了。 開篇說明 對的,讓你所見,又開始造輪子了。哈哈,造輪子我們是認真的~ 源碼閱讀是必須的,Underscore是因為剛剛學(xué)習(xí)整理了一波函數(shù)式編程,加上自己曾經(jīng)沒有太多閱讀源碼的經(jīng)驗,先拿Underscore練練手,...

    zorpan 評論0 收藏0
  • 1625行,解開 underscore.js 的面紗 - 第五章

    摘要:對多個一維數(shù)組進行并運算,實際上就是加強版的。所以我要說的是這個函數(shù),將傳入?yún)?shù)轉(zhuǎn)換為一個數(shù)組進行到的回調(diào)函數(shù)中,以此達到函數(shù)接到的是一個一維數(shù)組的集合。 每次小章節(jié)的開題都煩惱寫什么好,所以直接接下文 (~o▔▽▔)~o o~(▔▽▔o~) 。 _.first = _.head = _.take = function(array, n, guard) { if (arra...

    Rango 評論0 收藏0
  • 1625行,解開 underscore.js 的面紗 - 第六章

    摘要:用來構(gòu)成和兩個函數(shù),主要針對的是為了將函數(shù)調(diào)用模式更改為構(gòu)造器調(diào)用和方法調(diào)用。通過函數(shù)設(shè)定時間為毫秒后執(zhí)行函數(shù)的回調(diào)函數(shù),用以達到在規(guī)定時間毫秒時執(zhí)行函數(shù)的目的,并且規(guī)定時間內(nèi)只執(zhí)行一次函數(shù)。 北京的雨已經(jīng)斷斷續(xù)續(xù)下了好久,昏昏欲睡的躲在家里不愿意出門,火影忍者快要結(jié)束了,一拳超人第二季據(jù)說還要等好多年,勇者大冒險貌似斷更了,我又是在不喜歡海賊王的畫風(fēng),所以,我該看什么好呢。 va...

    v1 評論0 收藏0
  • 1625行,解開 underscore.js 的面紗 - 第二章

    摘要:第四個判斷如果是對象執(zhí)行返回一個斷言函數(shù),用來判定傳入對象是否匹配指定鍵值屬性。都不匹配最后執(zhí)行,返回傳入的對象的屬性。設(shè)置的值并生成函數(shù),等同于,使具有屬性且有值則返回,否則返回,這是一個判斷函數(shù)。 在第二小章節(jié)里面我按照源碼順序介紹幾個方法,源碼緊接著第一章繼續(xù): var builtinIteratee; builtinIteratee,內(nèi)置的 Iteratee (迭代器)。...

    yuxue 評論0 收藏0

發(fā)表評論

0條評論

zzzmh

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<