摘要:與最后,使用我們的寫的函數(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)); // 3restArgs
如果不使用 ... 拓展操作符,僅用 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)); // 6underscore 系列
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
摘要:所以它與其他系列的文章并不沖突,完全可以在閱讀完這個系列后,再跟著其他系列的文章接著學(xué)習(xí)。如何閱讀我在寫系列的時候,被問的最多的問題就是該怎么閱讀源碼我想簡單聊一下自己的思路。感謝大家的閱讀和支持,我是冴羽,下個系列再見啦 前言 別名:《underscore 系列 8 篇正式完結(jié)!》 介紹 underscore 系列是我寫的第三個系列,前兩個系列分別是 JavaScript 深入系列、...
摘要:他指示了一個對象的屬性,返回的將用來獲得該屬性對應(yīng)的值在上面的分析中,我們知道,當(dāng)傳入的是一個函數(shù)時,還要經(jīng)過一個叫的內(nèi)置函數(shù)才能獲得最終的所以此處的必然是優(yōu)化回調(diào)的作用了。 開篇說明 對的,讓你所見,又開始造輪子了。哈哈,造輪子我們是認真的~ 源碼閱讀是必須的,Underscore是因為剛剛學(xué)習(xí)整理了一波函數(shù)式編程,加上自己曾經(jīng)沒有太多閱讀源碼的經(jīng)驗,先拿Underscore練練手,...
摘要:對多個一維數(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...
摘要:用來構(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...
摘要:第四個判斷如果是對象執(zhí)行返回一個斷言函數(shù),用來判定傳入對象是否匹配指定鍵值屬性。都不匹配最后執(zhí)行,返回傳入的對象的屬性。設(shè)置的值并生成函數(shù),等同于,使具有屬性且有值則返回,否則返回,這是一個判斷函數(shù)。 在第二小章節(jié)里面我按照源碼順序介紹幾個方法,源碼緊接著第一章繼續(xù): var builtinIteratee; builtinIteratee,內(nèi)置的 Iteratee (迭代器)。...
閱讀 2158·2023-04-26 00:00
閱讀 3263·2021-09-24 10:37
閱讀 3534·2021-09-07 09:58
閱讀 1525·2019-08-30 15:56
閱讀 2223·2019-08-30 13:11
閱讀 2316·2019-08-29 16:38
閱讀 967·2019-08-29 12:58
閱讀 1884·2019-08-27 10:54