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

資訊專欄INFORMATION COLUMN

探究underscore源碼(一)

CloudwiseAPM / 2659人閱讀

摘要:什么鬼結(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是什么鬼?globalthis都能夠猜出來是全局變量,這個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];
    };
  };

這個方法淺顯易懂,如果傳入的objectnull,則返回 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

相關(guān)文章

  • 探究underscore源碼(二)

    摘要:源碼如下通過來判斷到底通過來區(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...

    jeyhan 評論0 收藏0
  • JavaScript專題系列20篇正式完結(jié)!

    摘要:寫在前面專題系列是我寫的第二個系列,第一個系列是深入系列。專題系列自月日發(fā)布第一篇文章,到月日發(fā)布最后一篇,感謝各位朋友的收藏點贊,鼓勵指正。 寫在前面 JavaScript 專題系列是我寫的第二個系列,第一個系列是 JavaScript 深入系列。 JavaScript 專題系列共計 20 篇,主要研究日常開發(fā)中一些功能點的實現(xiàn),比如防抖、節(jié)流、去重、類型判斷、拷貝、最值、扁平、柯里...

    sixleaves 評論0 收藏0
  • javascript知識點

    摘要:模塊化是隨著前端技術(shù)的發(fā)展,前端代碼爆炸式增長后,工程化所采取的必然措施。目前模塊化的思想分為和。特別指出,事件不等同于異步,回調(diào)也不等同于異步。將會討論安全的類型檢測惰性載入函數(shù)凍結(jié)對象定時器等話題。 Vue.js 前后端同構(gòu)方案之準(zhǔn)備篇——代碼優(yōu)化 目前 Vue.js 的火爆不亞于當(dāng)初的 React,本人對寫代碼有潔癖,代碼也是藝術(shù)。此篇是準(zhǔn)備篇,工欲善其事,必先利其器。我們先在代...

    Karrdy 評論0 收藏0
  • 你要看看這些有趣的函數(shù)方法嗎?

    前言 這是underscore.js源碼分析的第六篇,如果你對這個系列感興趣,歡迎點擊 underscore-analysis/ watch一下,隨時可以看到動態(tài)更新。 下劃線中有非常多很有趣的方法,可以用比較巧妙的方式解決我們?nèi)粘I钪杏龅降膯栴},比如_.after,_.before,_.defer...等,也許你已經(jīng)用過他們了,今天我們來深入源碼,一探究竟,他們到底是怎么實現(xiàn)的。 showIm...

    melody_lql 評論0 收藏0

發(fā)表評論

0條評論

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