成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

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

zzzmh / 2503人閱讀

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

partial

在《 JavaScript 專題之偏函數(shù)》中,我們寫(xiě)了一個(gè) partial 函數(shù),用來(lái)固定函數(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)語(yǔ)法,允許我們將一個(gè)不定數(shù)量的參數(shù)表示為一個(gè)數(shù)組。

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

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

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

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

寫(xiě)個(gè) demo,測(cè)試一下:

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

var addOne = partial(add, 1);

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

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

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

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

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

我們來(lái)寫(xiě)一版:

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

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

        // 使用一個(gè)數(shù)組儲(chǔ)存剩余的參數(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)化

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

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

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

func(1, 2)

所以我們?cè)賹?xiě)一版:

// 第二版
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)化

如果是正常寫(xiě)業(yè)務(wù),可能寫(xiě)到這里就結(jié)束了,然而 underscore 考慮的更多,鑒于 call 的性能要高于 apply,所以 underscore 做了一個(gè)優(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 的時(shí)候,我們可以以 _.restArgs 的形式調(diào)用該函數(shù)。

restArgs 與 partial

最后,使用我們的寫(xiě)的 restArgs 函數(shù)重寫(xiě)下 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ù)計(jì)寫(xiě)八篇左右,重點(diǎn)介紹 underscore 中的代碼架構(gòu)、鏈?zhǔn)秸{(diào)用、內(nèi)部函數(shù)、模板引擎等內(nèi)容,旨在幫助大家閱讀源碼,以及寫(xiě)出自己的 undercore。

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

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

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/90212.html

相關(guān)文章

  • underscore 的源碼該如何閱讀?

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

    weknow619 評(píng)論0 收藏0
  • 窺探Underscore源碼系列-開(kāi)篇

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

    zorpan 評(píng)論0 收藏0
  • 1625行,解開(kāi) underscore.js 的面紗 - 第五章

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

    Rango 評(píng)論0 收藏0
  • 1625行,解開(kāi) underscore.js 的面紗 - 第六章

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

    v1 評(píng)論0 收藏0
  • 1625行,解開(kāi) underscore.js 的面紗 - 第二章

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

    yuxue 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

zzzmh

|高級(jí)講師

TA的文章

閱讀更多
最新活動(dòng)
閱讀需要支付1元查看
<