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

資訊專欄INFORMATION COLUMN

Arale源碼解析(3)——Base模塊和Aspect模塊

stdying / 479人閱讀

摘要:本文同步自我的博客前言這個模塊實際上才是模塊系統(tǒng)中對外的模塊,它包含了之前介紹的類和類,以及自己內(nèi)部的模塊和模塊,因此模塊是真正的基礎類。這兩個方法的作用就是針對類上的某個方法,給這個方法綁定先于其執(zhí)行和后于其執(zhí)行的回調(diào)函數(shù)。

本文同步自我的GitHub博客

前言

Base這個模塊實際上才是Arale模塊系統(tǒng)中對外的模塊,它包含了之前介紹的Class類和Events類,以及自己內(nèi)部的attribute模塊和aspect模塊,因此Base模塊是真正的基礎類。

由于Attribute模塊的內(nèi)容太多,而Aspect模塊和它關系也不太大,因此,考慮到文章篇幅的平衡,將Base模塊的解析分成兩篇,Attribute模塊的分析放在下一篇多帶帶來寫。

帶注釋源碼

Base源碼的開頭是這樣的:

var Class = require("arale-class");
var Events = require("arale-events");
var Aspect = require("./aspect");
var Attribute = require("./attribute");

可見,整個Base的實現(xiàn)是基于上面這四個模塊的,前兩個模塊已經(jīng)分析過了,下面來分析后面兩個模塊。首先是Aspect模塊,這個模塊實際上只提供了兩個方法beforeafter

// `before`和`after`實際上是對`weave`方法的一次封裝,提供易用的接口
// 在指定方法執(zhí)行前,先執(zhí)行 callback
exports.before = function(methodName, callback, context) {
  return weave.call(this, "before", methodName, callback, context);
};

// 在指定方法執(zhí)行后,再執(zhí)行 callback
exports.after = function(methodName, callback, context) {
  return weave.call(this, "after", methodName, callback, context);
};

// Helpers
// -------
// 事件分割
var eventSplitter = /s+/;

/**
 * 控制callback的執(zhí)行時序
 * @param  {String}   when       選擇是`before`還是`after`
 * @param  {String}   methodName 方法名字符串
 * @param  {Function} callback   回調(diào)函數(shù)
 * @param  {Object}   context    上下文對象
 * @return {Object}              調(diào)用此方法的對象
 */
function weave(when, methodName, callback, context) {
  // 取得方法名數(shù)組
  var names = methodName.split(eventSplitter);
  var name, method;

  // 遍歷方法名數(shù)組
  while (name = names.shift()) {
    // 取得方法函數(shù)
    method = getMethod(this, name);
    // 方法是否被改造過,如果沒有則進行改造
    if (!method.__isAspected) {
      wrap.call(this, name);
    }
    // 綁定一下事件
    this.on(when + ":" + name, callback, context);
  }

  return this;
}

/**
 * 取得對應名稱的方法
 * @param  {Object} host       調(diào)用對象
 * @param  {String} methodName 方法名稱
 * @return {Function}            方法函數(shù)
 */
function getMethod(host, methodName) {
  // 取得對象上對應的方法函數(shù)
  var method = host[methodName];
  // 如果方法不存在則報錯
  if (!method) {
    throw new Error("Invalid method name: " + methodName);
  }
  return method;
}

/**
 * [wrap description]
 * @param  {[type]} methodName [description]
 * @return {[type]}            [description]
 */
function wrap(methodName) {
  // 取得對象上的方法
  var old = this[methodName];

  // 對方法進行改造封裝
  // 改造過的方法執(zhí)行時,會先觸發(fā)"before:methodName"事件
  this[methodName] = function() {
    // 切分參數(shù)
    var args = Array.prototype.slice.call(arguments);
    // 在參數(shù)數(shù)組前添加一項"before:methodName"
    var beforeArgs = ["before:" + methodName].concat(args);

    // prevent if trigger return false
    // 先觸發(fā)`before:methodName`事件,如果存在回調(diào)函數(shù)隊列且執(zhí)行后返回false,則阻止進一步往下執(zhí)行
    if (this.trigger.apply(this, beforeArgs) === false) return;

    // 執(zhí)行原方法,保存返回值
    var ret = old.apply(this, arguments);

    // 構造參數(shù)數(shù)組,執(zhí)行`after:methodName`事件
    var afterArgs = ["after:" + methodName, ret].concat(args);
    this.trigger.apply(this, afterArgs);

    return ret;
  };

  // 修改方法是否被改造狀態(tài)屬性
  this[methodName].__isAspected = true;
}

然后是Base模塊,它集成了Event, AspectAttribute模塊的各種屬性,實際上是Arale類庫的一個入口模塊:

var Class = require("arale-class");
var Events = require("arale-events");
var Aspect = require("./aspect");
var Attribute = require("./attribute");


module.exports = Class.create({
  // 混入Events, Aspect, Attribute模塊的所有屬性
  Implements: [Events, Aspect, Attribute],

  // 所有用`Base.extend()`構建的類在初始化時都會調(diào)用的方法
  initialize: function(config) {
    this.initAttrs(config);

    // 將`this._onChangeAttr`注冊為`change:attr`事件的監(jiān)聽函數(shù)
    parseEventsFromInstance(this, this.attrs);
  },

  destroy: function() {
    // 卸載所有事件監(jiān)聽
    this.off();

    // 清除所有屬性
    for (var p in this) {
      if (this.hasOwnProperty(p)) {
        delete this[p];
      }
    }

    // destroy一次后this都被清除了,再次調(diào)用會報錯,因此生成一個空的destroy,該方法與主同在
    this.destroy = function() {};
  }
});

/**
 * 將`_onChangeAttr`方法注冊為`change:attr`事件的監(jiān)聽函數(shù)
 * @param  {Class} host  調(diào)用對象
 * @param  {Object} attrs 包含所有要注冊屬性的對象
 */
function parseEventsFromInstance(host, attrs) {
  for (var attr in attrs) {
    if (attrs.hasOwnProperty(attr)) { // 檢測attr是attrs的非繼承屬性
      var m = "_onChange" + ucfirst(attr);
      if (host[m]) {
        host.on("change:" + attr, host[m]);
      }
    }
  }
}

/**
 * 將首字母轉變?yōu)榇髮? * @param  {String} str 要處理的字符串
 * @return {String}     處理完的字符串
 */
function ucfirst(str) {
  return str.charAt(0).toUpperCase() + str.substring(1);
}

源碼分析 Aspect

Aspect模塊就是實現(xiàn)了兩個方法,beforeafter。這兩個方法的作用就是針對類上的某個方法,給這個方法綁定先于其執(zhí)行和后于其執(zhí)行的回調(diào)函數(shù)。

兩個方法實際上調(diào)用的都是同一個方法weave,只是將before和after作為參數(shù)傳入,在weaver方法中,對要進行beforeafter“偽事件”綁定的方法進行查找,找到后會檢測這個方法上是否有__isAspected屬性。這個屬性的作用是標示出此方法有沒有被進行過偽事件的“包裝”。

上一段連續(xù)提到兩次“偽事件”這個詞,它是我編出來的,表示的意思為before:methodNameafter:methodName這樣的事件并不能成為一個獨立的事件,而是依附于methodName這個原方法的。原來的事件執(zhí)行流程是這樣的。

event.trigger(eventName)  +------------+
------------------------->| someMethod |----------->被觸發(fā)執(zhí)行
                          +------------+

一旦在someMethod上注冊了afterbefore事件后,someMethod就會被封裝成一個新的函數(shù):

someMethod被封裝后生成的新wrappedMethod:
                                      |trigger()
                  +-------------------------------------------------------+
                  |wrappedMethod:     |觸發(fā)`before:method`事件             |
                  |                   |                                   |
                  |            +---------------+  return false +-----+    |
                  |            |  beforeMethod |-------------->| end |    |
                  |            +---------------+               +-----+    |
                  |                   |return true                        |
                  |                   |                                   |
                  |            +---------------+                          |
                  |            |    method     |                          |
                  |            +---------------+                          |
                  |                   |觸發(fā)`after:method`事件              |
                  |                   |                                   |
                  |            +---------------+                          |
                  |            |  afterMethod  |                          |
                  |            +---------------+                          |
                  +-------------------------------------------------------+

整個模塊的關鍵就在于wrap這個用來封裝方法的函數(shù)了,當然實現(xiàn)這一功能的也需要功能完備的Event模塊的支持。

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

轉載請注明本文地址:http://systransis.cn/yun/85373.html

相關文章

  • Arale源碼解析(1)——Class

    摘要:先來看源碼中,首先是做的是參數(shù)的處理工作,針對某些參數(shù)未傳的情況作了調(diào)整,最后達到的效果是的值為傳入的父類構造函數(shù),如果沒有,設為。下一個語句其作用是處理父類構造函數(shù)沒有修改的屬性值并且有方法的時候,在上調(diào)用方法。 本文同步自我的GitHub 概述 Arale是支付寶開發(fā)的一套基礎類庫,提供了一整套前端模塊架構,基于CMD規(guī)范,所有模塊均是以sea.js的標準進行開發(fā)。其開發(fā)過程借...

    _ivan 評論0 收藏0
  • js 支持 Aspect 切面編程

    摘要:在方法執(zhí)行后,再執(zhí)行函數(shù)函數(shù)在執(zhí)行時,接收的參數(shù)第一個是的返回值,之后的參數(shù)和傳給相同。的返回值源碼定義兩個出口定義一個可柯里化的函數(shù),柯里化成函數(shù)指向基于生成的類的實例,如上例的如果該函數(shù)是第一次切面化綁定,則包裝該函數(shù)。 系列文章:讀 arale 源碼之 class 篇 使用 Aspect,可以允許你在指定方法執(zhí)行的前后插入特定函數(shù) before object.before(me...

    zhaot 評論0 收藏0
  • Arale源碼解析(2)——Events

    摘要:帶注釋源碼用于分割事件名的正則,識別空格介紹使用方法,這個模塊可以混入任何對象之中,實現(xiàn)對自定義事件的資瓷將空格分割的事件綁定給對象,事件名為的話,事件回調(diào)函數(shù)在任何事件被觸發(fā)時都會調(diào)用。 帶注釋源碼 // Regular expression used to split event strings // 用于分割事件名的正則,識別空格 var eventSplitter = /s+...

    adie 評論0 收藏0
  • arale 源碼之 attribute 篇

    摘要:系列文章讀源碼之篇提供基本的屬性添加獲取移除等功能。判斷是否為等對象特性檢查閉包實現(xiàn)塊作用域,不污染全局變量。找這個屬性,若沒有則返回空對象執(zhí)行函數(shù),返回被修改的值。 系列文章:讀 arale 源碼之 class 篇 attributes 提供基本的屬性添加、獲取、移除等功能。它是與實例相關的狀態(tài)信息,可讀可寫,發(fā)生變化時,會自動觸發(fā)相關事件 先來了解一下 Attribute 模塊要實...

    Magicer 評論0 收藏0
  • minipack源碼解析以及擴展

    摘要:的變化利用進行前后端通知。例如的副作用,資源只有資源等等,仔細剖析還有很多有趣的點擴展閱讀創(chuàng)建熱更新流程本文示例代碼聯(lián)系我 前置知識 首先可能你需要知道打包工具是什么存在 基本的模塊化演變進程 對模塊化bundle有一定了解 了解babel的一些常識 對node有一定常識 常見的一些打包工具 如今最常見的模塊化構建工具 應該是webpack,rollup,fis,parcel等等各...

    tangr206 評論0 收藏0

發(fā)表評論

0條評論

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