摘要:什么鬼結(jié)合上面的函數(shù),貌似可以看到每次調(diào)用函數(shù)時都會判斷一次是否等于。主要原理是利用回調(diào)函數(shù)來處理調(diào)用方法傳入的參數(shù)。
源頭本文基于underscore v1.8.3版本
一直想學(xué)習(xí)一下類庫的源碼,jQuery剛剛看到選擇器那塊,直接被那一大塊正則搞懵逼了。經(jīng)過同事的推薦,選擇了underscore來作為類庫研究的起點。
閉包所有函數(shù)都在一個閉包內(nèi),避免污染全局變量,這沒什么特殊的,略過。。。
(function() { ... }());全局對象的獲取
先看下面一段代碼:
var root = typeof self == "object" && self.self === self && self || typeof global == "object" && global.global === global && global || this;
self是什么鬼?global跟this都能夠猜出來是全局變量,這個self從哪里冒出來的?
第一眼看到這樣的代碼很困惑,感覺壓根沒有頭緒。但是如果你打開chrome的控制臺,神奇的事情發(fā)生了
其實查看源碼的注釋,我們也能看出來這段代碼的作用:在不一樣的環(huán)境里面獲取當(dāng)前全局對象this
self | window:瀏覽器
global:服務(wù)端
this:某些虛擬機
為了壓縮所做的原型賦值源碼中有將對象的原型鏈賦值給一個變量的做法:
var ArrayProto = Array.prototype, ObjProto = Object.prototype; var SymbolProto = typeof Symbol !== "undefined" ? Symbol.prototype : null;
一開始我并沒明白這么做的優(yōu)勢,代碼不都一樣嗎?
參考注釋并且上網(wǎng)查資料才知道原因:為了壓縮
舉個例子,Array.prototype是沒有辦法經(jīng)過壓縮的,Array,prototype這些,如果改了,瀏覽器就無法識別這些字段了。
但經(jīng)過類似上面代碼的處理,ObjProto經(jīng)過壓縮就能變成變量a,那么原來的代碼就會變成a.xxx。
我們平常寫的代碼也可以進行類似上面的處理,只要代碼的復(fù)用超過兩次,就可以考慮將其賦值給一個變量了。
this值統(tǒng)一處理this在類庫中的應(yīng)用很廣泛,undersocre采用了一個內(nèi)部函數(shù)來處理this:
var optimizeCb = function(func, context, argCount) { if (context === void 0) return func; switch (argCount == null ? 3 : argCount) { case 1: return function(value) { return func.call(context, value); }; // The 2-parameter case has been omitted only because no current consumers // made use of it. case 3: return function(value, index, collection) { return func.call(context, value, index, collection); }; case 4: return function(accumulator, value, index, collection) { return func.call(context, accumulator, value, index, collection); }; } return function() { return func.apply(context, arguments); }; };
注意到上面的case語句沒有2的情況,看其注釋基本就能明白,這是因為沒有使用到2的情況。
上面函數(shù)的最后一個參數(shù)argCount是用來指定參數(shù)個數(shù):
接受單值的情況
已取消
迭代器函數(shù)
reduce函數(shù)
callback的統(tǒng)一處理var cb = function(value, context, argCount) { if (_.iteratee !== builtinIteratee) return _.iteratee(value, context); if (value == null) return _.identity; if (_.isFunction(value)) return optimizeCb(value, context, argCount); if (_.isObject(value)) return _.matcher(value); return _.property(value); };
cb就是callback的簡寫,看函數(shù)的注釋的意思是:內(nèi)部函數(shù),用來生成可應(yīng)用于集合內(nèi)每個元素的回調(diào)函數(shù),返回預(yù)期的結(jié)果,具體應(yīng)用向下看。
iteratee什么鬼?_.iteratee = builtinIteratee = function(value, context) { return cb(value, context, Infinity); };
結(jié)合上面的cb函數(shù),貌似可以看到每次調(diào)用cb函數(shù)時都會判斷一次_.iteratee是否等于builtinIteratee。
如果不等于則調(diào)用_.iteratee函數(shù),讓_.iteratee = builtinIteratee,再繼續(xù)執(zhí)行cb函數(shù)。
結(jié)合注釋,猜測這個函數(shù)的作用應(yīng)該是防止用戶自己定義iteratee函數(shù)。
restArgs又一個基礎(chǔ)函數(shù)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); }; };
這個函數(shù)作用就類似ES6里面的rest params,這個函數(shù)主要是在官網(wǎng)分類里面的collections用到,例如:invoke。
主要原理是利用回調(diào)函數(shù)來處理調(diào)用方法傳入的參數(shù)。
創(chuàng)建繼承函數(shù)var baseCreate = function(prototype) { if (!_.isObject(prototype)) return {}; if (nativeCreate) return nativeCreate(prototype); Ctor.prototype = prototype; var result = new Ctor; // 創(chuàng)建 result 之后清空 Ctor 的原型鏈,防止 全局變量 Ctor 的原型鏈污染 Ctor.prototype = null; return result; };
主要原理就是利用 Ctor 做一個中介,創(chuàng)建繼承函數(shù)并返回后再清空Ctor的原型鏈,防止原型鏈污染
取對象的屬性值var property = function(key) { return function(obj) { return obj == null ? void 0 : obj[key]; }; };
這個方法淺顯易懂,如果傳入的object為null,則返回 undefined,否則返回屬性值。
其它的全局變量var MAX_ARRAY_INDEX = Math.pow(2, 53) - 1; var getLength = property("length"); var isArrayLike = function(collection) { var length = getLength(collection); return typeof length == "number" && length >= 0 && length <= MAX_ARRAY_INDEX; };
主要是幫助collection的方法來判定某個變量是否為collection。
后記看完這一段,感覺依舊有許多疑問。主要是因為這些全局定義的變量的使用場景沒有深究,更直白一些,就是沒有按照代碼的線索專研下去。希望在接下來的主要API的分析中能夠在好好回顧上面的那些函數(shù)以及變量
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/90959.html
摘要:源碼如下通過來判斷到底通過來區(qū)分對象以及數(shù)組。傳入回調(diào)函數(shù)的參數(shù)分別為對象鍵值對中的值或者數(shù)組中的序號值對象鍵值對中的鍵或者數(shù)組中的相應(yīng)序號舉個例子,傳入回調(diào)的參數(shù)依次為如果是數(shù)組,則傳入?yún)?shù)依次為三這幾個方法都是利用一個核心函數(shù)。 一、_.each 一開始我并沒有以為_.each這個方法會有多大的用處,不就是一個遍歷嘛~ 但當(dāng)我利用自己測試這個函數(shù)的時候,發(fā)現(xiàn)了一件大事 under...
摘要:寫在前面專題系列是我寫的第二個系列,第一個系列是深入系列。專題系列自月日發(fā)布第一篇文章,到月日發(fā)布最后一篇,感謝各位朋友的收藏點贊,鼓勵指正。 寫在前面 JavaScript 專題系列是我寫的第二個系列,第一個系列是 JavaScript 深入系列。 JavaScript 專題系列共計 20 篇,主要研究日常開發(fā)中一些功能點的實現(xiàn),比如防抖、節(jié)流、去重、類型判斷、拷貝、最值、扁平、柯里...
摘要:模塊化是隨著前端技術(shù)的發(fā)展,前端代碼爆炸式增長后,工程化所采取的必然措施。目前模塊化的思想分為和。特別指出,事件不等同于異步,回調(diào)也不等同于異步。將會討論安全的類型檢測惰性載入函數(shù)凍結(jié)對象定時器等話題。 Vue.js 前后端同構(gòu)方案之準(zhǔn)備篇——代碼優(yōu)化 目前 Vue.js 的火爆不亞于當(dāng)初的 React,本人對寫代碼有潔癖,代碼也是藝術(shù)。此篇是準(zhǔn)備篇,工欲善其事,必先利其器。我們先在代...
前言 這是underscore.js源碼分析的第六篇,如果你對這個系列感興趣,歡迎點擊 underscore-analysis/ watch一下,隨時可以看到動態(tài)更新。 下劃線中有非常多很有趣的方法,可以用比較巧妙的方式解決我們?nèi)粘I钪杏龅降膯栴},比如_.after,_.before,_.defer...等,也許你已經(jīng)用過他們了,今天我們來深入源碼,一探究竟,他們到底是怎么實現(xiàn)的。 showIm...
閱讀 664·2021-11-15 11:39
閱讀 2901·2021-10-08 10:04
閱讀 3265·2019-08-30 10:57
閱讀 3025·2019-08-26 13:25
閱讀 1908·2019-08-26 12:14
閱讀 2636·2019-08-23 15:27
閱讀 2996·2019-08-23 15:18
閱讀 1777·2019-08-23 14:26