精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

大型JavaScript應用程序架構模式

開發 前端
11月中旬在倫敦舉行的jQuery Summit頂級大會上有個session講的是大型JavaScript應用程序架構,看完PPT以后覺得甚是不錯,于是整理一下發給大家共勉。

PDF版的PPT下載地址:http://www.slideshare.net/jibyjohnc/jqquerysummit-largescale-javascript-application-architecture

注:在整理的過程中,發現作者有些思想是返來復去地說,所以刪減了一部分,如果你的英文良好,請直接閱讀英文的PPT。

什么叫“JavaScript大型程序”?

在我們開始之前,我們來定義一下什么叫大型JavaScript站點,很多有經驗的JS開發高手也都被challenge住了,有人說超過10萬行JavaScript代碼才算大型,也有人說JavaScript代碼要超過1MB大小才算,其實2者都不能算對,因為不能安裝代碼量的多少來衡量,很多瑣碎的JS代碼很容易超過10萬行的。

我對“大”的定義如下,雖然可能不太對,但是應該是比較接近了:

我個人認為,大型JavaScript程序應該是非常重要并且融入了很多卓越開發人員努力,對重量級數據進行處理并且展示給瀏覽器的程序。

回顧當前的程序架構

我不能強調說這個問題有多重要,很多有經驗的開發人員經常說:“現有的創意和設計模式在我上一個中型項目上運行得非常好,所以在稍微大型點的程序里再次使用,應該沒問題,對吧?”,在一定程序上是沒錯的,但別忘記了,既然是大型程序,通常就應該有大的Concerns需要分解關注,我簡短解釋一下要花時間來review當前運行了很久的程序架構。大多數情況下,當前的JavaScript程序架構應該是如下這個樣子的(注意,是JS架構,不是大家常說的ASP.NET MVC):

custom widgets

models

views

controllers

templates

libraries/toolkits

an application core.

你可能還會將程序單獨封裝成多個modules,或者使用其他的設計模式,這很好,但是如果這些結構完全代表你的架構的話,就可能會有一些潛在的問題,我們來看看幾個重要的點:

1.你架構里的東西,有多少可以立即拿出來重用?

有沒有一些單獨的module不依賴別的代碼?是自包含么?如果我到你們正在使用的代碼庫上去隨即挑選一些模塊module代碼,然后放在一個新頁面,是否能立即就能使用?你可能會說原理通就可以了,我建議你長久打算一下,如果你的公司之前開發很多重要的程序,突然有一天有人說,這個項目里的聊天模塊不錯,我們拿出來放在另外一個項目里吧,你能直接拿過來不修改代碼就能使用么?

2.系統里有多少模塊module需要依賴其他模塊?

系統的各個模塊是不是都很緊耦合?在我將這個問題作為concern之前,我先解釋一下,不是說所有的模塊都絕對不能有任何依賴,比如一個細粒度的功能可能是從base功能擴展來的,我的問題和這種情況不一樣,我說的是不同功能模塊之前的依賴,理論上,所有的不同功能模塊都不應該有太多的依賴。

3.如果你程序的某一部分出錯了,其他部分是否能夠依然工作?

如果你構建一個和Gmail差不多的程序,你可以發現Gmail里很多模塊都是動態加載的,比如聊天chat模塊,在初始化頁面的時候是不加載的,而且就算加載以后出錯了,頁面的其他部分也能正常使用。

4.你的各個模塊Module能很簡單的進行測試么?

你的每一個模塊都有可能用在數百萬用戶的大型站點上,甚至多個站點都使用它,所以你的模塊需要能經得住測試,也就是說,不管是在架構內部還是架構外部,都應該能很簡單的去測試,包括大部分的斷言在不同的環境下都能夠通過。

長遠考慮

架構大型程序的時候,最重要的是要有前瞻性,不能只考慮一個月或者一年以后的情況,要考慮更長久的情況下,有什么改變的可能性?開發人員經常將DOM操作的代碼和程序綁定得太緊,盡管有時候已經封裝單獨的邏輯到不同的模塊里了,想想一下,長久以后,為什么不是很好。

我的一個同事曾經說過,一個精確的架構可能不適合未來的情景,有時候是正確的,但是當你需要該做的話,你所付出的money那可是相當地多哦。比如,你可能因為某些性能,安全,設計的原因需要在Dojo, jQuery, Zepto, YUI之間需要選擇替換,這時候就有問題了,大部分模塊都有依賴,需要錢呀,需要時間啊,需要人呀,對不?

對于一些小型站點沒事,但是大型站點確實需要提供一個更加靈活的機制,而不去擔心各個模塊之間的各種問題,這既然節約錢,又能節約時間。

總結一下,現在你能確定你能不重寫整個程序就能替換一些類庫么?如果不能,那估計我們下面要講的內容,就比較適合你了。

很多有經驗的JavaScript開發者給出了一些關鍵的notes:

JavaScriptMVC的作者Justin Meyer說:

構建大型程序最大的秘密就是從來不構建大型程序,而是將程序分解成各個小的模塊去做,讓每個小模塊都可測試,可size化,然后集成到程序里。

High-performance JavaScript websites作者Nicholas,Zakas:

"The key is to acknowledge from the start that you have no idea how this will grow. When you accept that you don't know everything, you begin to design the system defensively. You identify the key areas that may change, which often is very easy when you put a little bit of time into it. For instance, you should expect that any part of the app that communicates with another system will likely change, so you need to abstract that away." -

一大堆文字問題,太麻煩了,總結一句就是,一切皆可變,所以要抽象。

jQuery Fundamentals作者Rebecca Murphey:

各個模塊之間聯系的越密切,重用性越小,改變起來困難越大。

以上這些重要觀點,是構建架構的核心要素,我們需要時刻銘記。

頭腦風暴

我們來頭腦風暴一下,我們需要一個松耦合的架構,各模塊之間沒有依賴,各個模塊和程序進行通信,然后中間層接管和處理反饋相應的消息。

例如,我們如果有一個JavaScript構建在線面包店程序,一個模塊發出了一個信息可能是“有42個圓面包需要派件”。我們使用不同的layer層來處理模塊發來的消息,做到如下:

◆ 模塊不直接訪問程序核心

◆ 模塊不直接調用或影響其它的模塊

這將防止我們因為某個模塊出錯,而導致所有的模塊出錯。

另外一個問題是安全,真實的情況是,大多數人都不認為內部安全是個問題,我們自己心里說,程序是我自己構建的,我知道哪些是公開的那些私有的,安全沒問題,但你有沒有辦法去定義哪個模塊才能權限訪問程序核心?例如,有一個chat聊天模塊,我不想讓他調用admin模塊,或者不想讓它調用有DB寫權限的模塊,因為這之間存在很脆弱,很容易導致XSS攻擊。每個模塊不應該能做所有的事情,但是當前大多數架構里的JavaScript代碼都有這種的問題。提供一個中間層來控制,哪個模塊可以訪問那個授權的部分,也就是說,該模塊最多只能做到我們所授權的那部分。

#p#

建議的架構

我們本文的重點來了,這次我們提議的架構使用了我們都很熟知的設計模式:module, facade和mediator。

和傳統的模型不一樣的是,為了解耦各個模塊,我們只讓模塊發布一些event事件,mediator模式可以負責從這些模塊上訂閱消息message,然后控制通知的response,facade模式用戶限制各模塊的權限。

以下是我們要注意講解的部分:

1 設計模式

1.1 模塊論

1.1.1 綜述

1.1.2 Module模式

1.1.3 對象自面量

1.1.4 CommonJS模塊

1.2 Facade模式

1.3 Mediator模式

2 應用到你的架構

2.1 Facade - 核心抽象

2.2 Mediator - 程序核心

2.3 緊密聯合運作起來

模塊論

大家可能都或多或少地使用了模塊化的代碼,模塊是一個完整的強健程序架構的一部分,每個模塊都是為了單獨的目的為創建的,回到Gmail,我們來個例子,chat聊天模塊看起來是個單獨的一部分,其實它是有很多單獨的子模塊來構成,例如里面的表情模塊其實就是單獨的子模塊,也被用到了發送郵件的窗口上。

另外一個是模塊可以動態加載,刪除和替換。

在JavaScript里,我們又幾種方式來實現模塊,大家熟知的是module模式和對象字面量,如果你已經熟悉這些,請忽略此小節,直接跳到CommonJS部分。

Module模式

module模式是一個比較流行的設計模式,它可以通過大括號封裝私有的變量,方法,狀態的,通過包裝這些內容,一般全局的對象不能直接訪問,在這個設計模式里,只返回一個API,其它的內容全部被封裝成私有的了。

另外,這個模式和自執行的函數表達式比較相似,唯一的不同是module返回的是對象,而自執行函數表達式返回的是function。

眾所周知, JavaScript不想其它語言一樣有訪問修飾符,不能為每個字段或者方法聲明private,public修飾符,那這個模式我們是如何實現的呢?那就是return一個對象,里面包括一些公開的方法,這些方法有能力去調用內部的對象。

看一下,下面的代碼,這段代碼是一個自執行代碼,聲明里包括了一個全局的對象basketModule, basket數組是一個私有的,所以你的整個程序是不能訪問這個私有數組的,同時我們return了一個對象,其內包含了3個方法(例如addItem,getItemCount,getTotal),這3個方法可以訪問私有的basket數組。

  1. var basketModule = (function() {  
  2. var basket = []; //private  
  3. return { //exposed to public  
  4.   addItem: function(values) {  
  5.     basket.push(values);  
  6.   },  
  7.   getItemCount: function() {  
  8.     return basket.length;  
  9.   },  
  10.   getTotal: function(){  
  11.     var q = this.getItemCount(),p=0;  
  12.     while(q--){  
  13.     p+= basket[q].price;  
  14.     }  
  15.     return p;  
  16.   }  
  17.  }  
  18. }()); 

同時注意,我們return的對象直接賦值給了basketModule,所以我們可以像下面一樣使用:

  1. //basketModule is an object with properties which can also be methods  
  2. basketModule.addItem({item:'bread',price:0.5});  
  3. basketModule.addItem({item:'butter',price:0.3});  
  4.    
  5. console.log(basketModule.getItemCount());  
  6. console.log(basketModule.getTotal());  
  7.    
  8. //however, the following will not work:  
  9. console.log(basketModule.basket);// (undefined as not inside the returned object)  
  10. console.log(basket); //(only exists within the scope of the closure) 

那在各個流行的類庫(如Dojo, jQuery)里是如何來做呢?

Dojo

Dojo試圖使用dojo.declare來提供class風格的聲明方式,我們可以利用它來實現Module模式,例如如果你想再store命名空間下聲明basket對象,那么可以這么做:

  1. //traditional way  
  2.   var store = window.store || {};  
  3.   store.basket = store.basket || {};  
  4.      
  5.   //using dojo.setObject  
  6.   dojo.setObject("store.basket.object", (function() {  
  7.     var basket = [];  
  8.     function privateMethod() {  
  9.       console.log(basket);  
  10.     }  
  11.     return {  
  12.       publicMethod: function(){  
  13.         privateMethod();  
  14.       }  
  15.      };  
  16.   }())); 

結合dojo.provide一起來使用,非常強大。

YUI

下面的代碼是YUI原始的實現方式:

  1. YAHOO.store.basket = function () {  
  2.  
  3.     //"private" variables:  
  4.     var myPrivateVar = "I can be accessed only within YAHOO.store.basket .";  
  5.  
  6.     //"private" method:  
  7.     var myPrivateMethod = function () {  
  8.         YAHOO.log("I can be accessed only from within YAHOO.store.basket");  
  9.     }  
  10.  
  11.     return {  
  12.         myPublicProperty: "I'm a public property.",  
  13.         myPublicMethod: function () {  
  14.             YAHOO.log("I'm a public method.");  
  15.  
  16.             //Within basket, I can access "private" vars and methods:  
  17.             YAHOO.log(myPrivateVar);  
  18.             YAHOO.log(myPrivateMethod());  
  19.  
  20.             //The native scope of myPublicMethod is store so we can  
  21.             //access public members using "this":  
  22.             YAHOO.log(this.myPublicProperty);  
  23.         }  
  24.     };  
  25.  
  26. } (); 

jQuery

jQuery里有很多Module模式的實現,我們來看一個不同的例子,一個library函數聲明了一個新的library,然后創建該library的時候,在document.ready里自動執行init方法。

  1. function library(module) {  
  2.   $(function() {  
  3.     if (module.init) {  
  4.       module.init();  
  5.     }  
  6.   });  
  7.   return module;  
  8. }  
  9.    
  10. var myLibrary = library(function() {  
  11.   return {  
  12.     init: function() {  
  13.       /*implementation*/ 
  14.       }  
  15.   };  
  16. }()); 

#p#

對象自面量

對象自面量使用大括號聲明,并且使用的時候不需要使用new關鍵字,如果對一個模塊里的屬性字段的publice/private不是很在意的話,可以使用這種方式,不過請注意這種方式和JSON的不同。對象自面量:var item={name: "tom", value:123} JSON:var item={"name":"tom", "value":123}。

  1. var myModule = {  
  2.     myProperty: 'someValue',  
  3.     //object literals can contain properties and methods.  
  4.     //here, another object is defined for configuration  
  5.     //purposes:  
  6.     myConfig: {  
  7.         useCaching: true,  
  8.         language: 'en' 
  9.     },  
  10.     //a very basic method  
  11.     myMethod: function () {  
  12.         console.log('I can haz functionality?');  
  13.     },  
  14.     //output a value based on current configuration  
  15.     myMethod2: function () {  
  16.         console.log('Caching is:' + (this.myConfig.useCaching) ? 'enabled' : 'disabled');  
  17.     },  
  18.     //override the current configuration  
  19.     myMethod3: function (newConfig) {  
  20.         if (typeof newConfig == 'object') {  
  21.             this.myConfig = newConfig;  
  22.             console.log(this.myConfig.language);  
  23.         }  
  24.     }  
  25. };  
  26.  
  27. myModule.myMethod(); //I can haz functionality  
  28. myModule.myMethod2(); //outputs enabled  
  29. myModule.myMethod3({ language: 'fr', useCaching: false }); //fr 

CommonJS

關于 CommonJS的介紹,這里就不多說了,博客園有很多帖子都有介紹,我們這里要提一下的是CommonJS標準里里有2個重要的參數exports和require,exports是代表要加載的模塊,require是代表這些加載的模塊需要依賴其它的模塊,也需要將它加載進來。

  1. /*  
  2.    Example of achieving compatibility with AMD and standard CommonJS by putting boilerplate around the standard CommonJS module format:  
  3.    */ 
  4.       
  5.    (function(define){  
  6.      define(function(require,exports){  
  7.        // module contents  
  8.        var dep1 = require("dep1");  
  9.        exports.someExportedFunction = function(){...};  
  10.        //...  
  11.      });  
  12.    })(typeof define=="function"?define:function(factory){factory(require,exports)}); 

有很多CommonJS標準的模塊加載實現,我比較喜歡的是RequireJS,它能否非常好的加載模塊以及相關的依賴模塊,來一個簡單的例子,例如需要將圖片轉化成ASCII碼,我們先加載encoder模塊,然后獲取他的encodeToASCII方法,理論上代碼應該是如下:

  1. var encodeToASCII = require("encoder").encodeToASCII;  
  2.  exports.encodeSomeSource = function(){  
  3.    //其它操作以后,然后調用encodeToASCII  
  4.  } 

但是上述代碼并沒用工作,因為encodeToASCII函數并沒用附加到window對象上,所以不能使用,改進以后的代碼需要這樣才行:

  1. define(function(require, exports, module) {  
  2.      var encodeToASCII = require("encoder").encodeToASCII;  
  3.        exports.encodeSomeSource = function(){  
  4.        //process then call encodeToASCII  
  5.      }  
  6.    }); 

CommonJS 潛力很大,但是由于大叔不太熟,所以就不過多地介紹了。

Facade模式

Facade模式在本文架構里占有重要角色,關于這個模式很多JavaScript類庫或者框架里都有體現,其中最大的作用,就是包括High level的API,以此來隱藏具體的實現,這就是說,我們只暴露接口,內部的實現我們可以自己做主,也意味著內部實現的代碼可以很容易的修改和更新,比如今天你是用jQuery來實現的,明天又想換YUI了,這就非常方便了。

下面這個例子了,可以看到我們提供了很多私有的方法,然后通過暴露一個簡單的 API來讓外界執行調用內部的方法:

  1. var module = (function () {  
  2.     var _private = {  
  3.         i: 5,  
  4.         get: function () {  
  5.             console.log('current value:' + this.i);  
  6.         },  
  7.         set: function (val) {  
  8.             this.i = val;  
  9.         },  
  10.         run: function () {  
  11.             console.log('running');  
  12.         },  
  13.         jump: function () {  
  14.             console.log('jumping');  
  15.         }  
  16.     };  
  17.     return {  
  18.         facade: function (args) {  
  19.             _private.set(args.val);  
  20.             _private.get();  
  21.             if (args.run) {  
  22.                 _private.run();  
  23.             }  
  24.         }  
  25.     }  
  26. } ());  
  27.  
  28. module.facade({run:true, val:10});  
  29. //outputs current value: 10, running 

Facade和下面我們所說的mediator的區別是,facade只提供現有存在的功能,而mediator可以增加新功能。

#p#

Mediator模式

講modiator之前,我們先來舉個例子,機場飛行控制系統,也就是傳說中的塔臺,具有絕對的權利,他可以控制任何一架飛機的起飛和降落時間以及地方,而飛機和飛機之前不允許通信,也就是說塔臺是機場的核心,mediator就相當于這個塔臺。

mediator就是用在程序里有多個模塊,而你又不想讓各個模塊有依賴的話,那通過mediator模式可以達到集中控制的目的。實際場景中也是,mediator封裝了很多不想干的模塊,讓他們通過mediator聯系在一起,同時也松耦合他們,使得他們之間必須通過mediator才能通信。

那mediator模式的優點是什么?那就是解耦,如果你之前對觀察者模式比較了解的話,那理解下面的mediator圖就相對簡單多了,下圖是一個high level的mediator模式圖:

 

 

想想一下,各模塊是發布者,mediator既是發布者又是訂閱者。

Module 1向Mediator廣播一個實際,說需要做某事

Mediator捕獲消息以后,立即啟動處理該消息需要使用的Module 2,Module 2處理結束以后返回信息給Mediator

與此同時,Mediator也啟動了Module 3,當接受Module 2 返回消息的時候自動記錄日志到Module 3里

可以看到,各模塊之間并沒有通信,另外Mediator也可以實現監控各模塊狀態的功能,例如如果Module 3出錯了,Mediator可以暫時只想其它模塊,然后重啟Module 3,然后繼續執行。

回顧一下,可以看到,Mediator的優點是:松耦合的模塊由同一的Mediator來控制,模塊只需要廣播和監聽事件就可以了,而模塊之間不需要直接聯系,另外,一次信息的處理可以使用多個模塊,也方便我們以后統一的添加新的模塊到現有的控制邏輯里。

確定是:由于所有的模塊直接都不能直接通信,所有相對來說,性能方面可能會有少許下降,但是我認為這是值得的。

我們根據上面的講解來一個簡單的Demo:

  1. var mediator = (function(){  
  2.        var subscribe = function(channel, fn){  
  3.            if (!mediator.channels[channel]) mediator.channels[channel] = [];  
  4.            mediator.channels[channel].push({ context: this, callback: fn });  
  5.            return this;  
  6.        },  
  7.       
  8.        publish = function(channel){  
  9.            if (!mediator.channels[channel]) return false;  
  10.            var args = Array.prototype.slice.call(arguments, 1);  
  11.            for (var i = 0, l = mediator.channels[channel].length; i < l; i++) {  
  12.                var subscription = mediator.channels[channel][i];  
  13.                subscription.callback.apply(subscription.context, args);  
  14.            }  
  15.            return this;  
  16.        };  
  17.       
  18.        return {  
  19.            channels: {},  
  20.            publish: publish,  
  21.            subscribe: subscribe,  
  22.            installTo: function(obj){  
  23.                obj.subscribe = subscribe;  
  24.                obj.publish = publish;  
  25.            }  
  26.        };  
  27.       
  28.    }()); 

然后有2個模塊分別調用:

  1. //Pub/sub on a centralized mediator  
  2.       
  3.    mediator.name = "tim";  
  4.    mediator.subscribe('nameChange'function(arg){  
  5.            console.log(this.name);  
  6.            this.name = arg;  
  7.            console.log(this.name);  
  8.    });  
  9.       
  10.    mediator.publish('nameChange''david'); //tim, david  
  11.       
  12.       
  13.    //Pub/sub via third party mediator  
  14.       
  15.    var obj = { name: 'sam' };  
  16.    mediator.installTo(obj);  
  17.    obj.subscribe('nameChange'function(arg){  
  18.            console.log(this.name);  
  19.            this.name = arg;  
  20.            console.log(this.name);  
  21.    });  
  22.       
  23.    obj.publish('nameChange''john'); //sam, john 

應用Facade: 應用程序核心的抽象

一個facade是作為應用程序核心的一個抽象來工作的,在mediator和模塊之間負責通信,各個模塊只能通過這個facade來和程序核心進行通信。作為抽象的職責是確保任何時候都能為這些模塊提供一個始終如一的接口(consistent interface),和sendbox controller的角色比較類似。所有的模塊組件通過它和mediator通信,所以facade需要是可靠的,可信賴的,同時作為為模塊提供接口的功能,facade還需要扮演另外一個角色,那就是安全控制,也就是決定程序的哪個部分可以被一個模塊訪問,模塊組件只能調用他們自己的方法,并且不能訪問任何未授權的內容。例如,一個模塊可能廣播dataValidationCompletedWriteToDB,這里的安全檢查需要確保該模塊擁有數據庫的寫權限。

總之,mediator只有在facade授權檢測以后才能進行信息處理。

應用Mediator:應用程序的核心

Mediator是作為應用程序核心的角色來工作的,我們簡單地來說一下他的職責。最核心的工作就是管理模塊的生命周期(lifecycle),當這個核心撲捉到任何信息進來的時候,他需要判斷程序如何來處理——也就是說決定啟動或停止哪一個或者一些模塊。當一個模塊開始啟動的時候,它應該能否自動執行,而不需要應用程序核心來決定是否該執行(比如,是否要在DOM ready的時候才能執行),所以說需要模塊自身需要去判定。

你可能還有問題,就是一個模塊在什么情況下才會停止。當程序探測到一個模塊失敗了,或者是出錯了,程序需要做決定來防止繼續執行該模塊里的方法,以便這個組件可以重新啟動,目的主要是提高用戶體驗。

另外,該核心應該可以動態添加或者刪除模塊,而不影響其他任何功能。常見的例子是,一個模塊在頁面加載初期是不可用,但是用戶操作以后,需要動態加載這個模塊然后執行,就像Gmail里的chat聊天功能一樣,從性能優化的目的來看,應該是很好理解的吧。

異常錯誤處理,也是由應用程序核心來處理的,另外各模塊在廣播信息的時候,也廣播任何錯誤到該核心里,以便程序核心可以根據情況去停止/重啟這些模塊。這也是松耦合架構一個很重要的部分,我們不需要手工改變任何模塊,通過mediator使用發布/訂閱就可以來做到這個。

組裝起來

各模塊包含了程序里各種各樣的功能,他們有信息需要處理的時候,發布信息通知程序(這是他們的主要職責),下面的QA小節里提到了,模塊可以依賴一些DOM工具操作方法,但是不應該和系統的其它模塊有依賴,一個模塊不應該關注如下內容:

哪個對象或者模塊訂閱了這個模塊發布的信息

這些對象是客戶端對象還是服務器端對象

多少對象訂閱了你的信息

 

 

Facade抽象應用程序的核心,避免各個模塊之間直接通信,它從各模塊上訂閱信息,也負責授權檢測,確保每個模塊有用自己單獨的授權。

 

 

Mediator(應用程序核心)使用mediator模式扮演發布/訂閱管理器的角色,負責模塊管理以及啟動/停止模塊執行,可以動態加載以及重啟有錯誤的模塊。

 

 

這個架構的結果是:各模塊之間沒有依賴,因為松耦合的應用,它們可以很容易地被測試和維護,各模塊可以很容易地在其它項目里被重用,也可以在不影響程序的情況下動態添加和刪除。

發布Pub/訂閱Sub的延伸:自動注冊事件(Automatic Event Registration)

關于自動注冊事件,需要遵守一定的命名規范,比如如果一個模塊發布了一個名字為messageUpdate的事件,那么所有帶有messageUpdate方法的模塊都會被自動執行。有好處也有利弊,具體實現方式,可以看我另外一篇帖子:jQuery自定義綁定的魔法升級版。

QA

1.有可能不使用facade或者類似的sandbox模式么?

盡管架構的大綱里提出了facade可以實現授權檢查的功能,其實完全可能由mediator去做,輕型架構要做的事情其實是幾乎一樣的,那就是解耦,確保各模塊直接和應用程序核心通信是沒問題的就行。

2.你提高了模塊直接不能有依賴,是否意味著不能依賴任何第三方類庫(例如jQuery)。

這其實就是一個兩面性的問題,我們上面說到了,一個模塊也許有一些子模塊,或者基礎模塊,比如基本的DOM操作工具類等,在這個層面上講,我們是可以用第三方類庫的,但是請確保,我們可以很容易地能否替換掉他們。

3.我喜歡這個架構,并且想開始使用這個架構,有任何代碼樣本可以參考么?

我打算去搞一份代碼樣本供大家參考,不過在這之前,你可以參考Andrew Burgees的帖子Writing Modular JavaScript 。

4.如果模塊需要和應用程序核心直接通信,是否可行?

技術上來將,沒有理由現在模塊不能和應用程序核心直接通信,但是對于大多數應用體驗來說,還是不要。既然你選擇了這個架構,那就要遵守該架構所定義的規則。

原文:http://www.cnblogs.com/TomXu/archive/2011/12/14/2286225.html
 

【編輯推薦】

  1. Firefox 9發布 可提升JavaScript性能20-30%
  2. JavaScript開發者必備的十款超級有用的工具
  3. Google提供JavaScript庫以簡化Google API的調用
  4. 如何實現高效的 JavaScript 單元測試
  5. JavaScript異步編程之Jscex版火拼俄羅斯
責任編輯:陳貽新 來源: 湯姆大叔的博客
相關推薦

2012-01-13 10:45:56

企業應用架構

2010-09-30 13:39:52

2012-06-07 09:15:14

ibmdw

2012-10-08 11:16:36

企業應用架構標識映射

2024-06-04 00:00:04

2012-10-08 11:18:38

企業應用架構工作單元模式

2009-04-16 08:05:27

Windows Mob微軟移動OS

2011-02-22 10:23:43

2023-06-16 13:34:00

軟件架構模式

2009-04-15 09:07:45

微軟Windows Mob在線程序商店

2024-02-05 08:11:30

架構模式單體

2012-01-13 10:31:25

ibmdw

2010-01-04 10:41:14

Silverlight

2011-08-10 11:25:59

ipad信息架構數據結構

2011-07-22 14:08:19

iPad 架構 數據

2023-01-09 17:04:24

2015-05-04 14:07:54

2024-03-26 13:35:19

模型架構框架

2024-09-02 10:07:52

2020-12-29 19:15:04

架構模式代碼分布式
點贊
收藏

51CTO技術棧公眾號

日韩中文字幕组| 国产精品一 二 三| 日本精品人妻无码77777| baoyu135国产精品免费| 色综合一个色综合亚洲| 亚洲图片小说在线| 丰满人妻一区二区三区四区53| 国产亚洲精品久久久久婷婷瑜伽| 中文字幕精品—区二区| 18禁一区二区三区| 3d欧美精品动漫xxxx无尽| 亚洲人精品午夜| 久久综合色一本| 国产男女裸体做爰爽爽| 性欧美长视频| 色综合视频网站| 亚洲高潮女人毛茸茸| 澳门成人av| 欧美精品日日鲁夜夜添| 一本大道熟女人妻中文字幕在线 | 久久精品国产欧美激情| 亚洲观看黄色网| 亚洲人成777| 色av成人天堂桃色av| 996这里只有精品| 免费观看久久久久| 91麻豆精品秘密| 成人av资源| 国产日韩在线观看一区| 日韩精品一卡二卡三卡四卡无卡| 欧美精品第一页在线播放| 国产黄色片在线| 国产一区网站| 日韩精品中文字幕有码专区| 国产伦理在线观看| 久久中文字幕一区二区| 欧美日韩精品一二三区| 成年人小视频网站| 成人美女大片| 高潮白浆女日韩av免费看| 少妇久久久久久被弄到高潮| 美女av在线播放| 中文字幕av一区二区三区| 欧美日韩成人一区二区三区| 神马久久久久久久久久| 国产不卡在线视频| dy888夜精品国产专区| 国产日韩欧美中文字幕| 国产在线精品免费av| 国产一区二区在线免费| 中文字幕在线网址| 麻豆国产欧美日韩综合精品二区 | 91成人app| 欧美男女性生活在线直播观看 | 国产一区二区不卡在线| 成人美女免费网站视频| 国产精品久久久久久久免费| 精品一区二区三区av| 成人精品视频在线| www.天天干.com| 成人久久视频在线观看| 国产区一区二区三区| 视频国产一区二区三区| 久久综合资源网| 任我爽在线视频精品一| 95在线视频| 亚洲天堂免费看| 日本三级中文字幕在线观看| 欧美黑人猛交| 欧美丝袜第一区| 免费国产成人av| 亚洲免费资源| 亚洲成人精品视频在线观看| 国产网站无遮挡| 日本一区二区三区视频| 久久综合伊人77777蜜臀| 九九九国产视频| 免费亚洲一区| 成人h片在线播放免费网站| 亚洲第一天堂网| 91社区在线播放| 亚洲欧美久久234| 久久香蕉一区| 91福利精品视频| 日本亚洲一区二区三区| 大香伊人久久精品一区二区 | asian性开放少妇pics| 欧美日韩在线二区| 久精品免费视频| 在线天堂中文字幕| 老司机午夜精品| 国产精品免费视频一区二区| 黄色片在线免费看| 亚洲综合色噜噜狠狠| 日韩 欧美 高清| 久久99成人| 亚洲品质视频自拍网| 91麻豆免费视频网站| 国产一区二区三区的电影| 国产精品成人久久久久| 亚洲精品久久久久久动漫器材一区| 91免费看视频| 男人的天堂视频在线| 日韩国产网站| 精品国产3级a| 男人晚上看的视频| 久久一区激情| 国产91一区二区三区| 国产黄色免费在线观看| 亚洲国产成人av网| 亚洲男人天堂2021| 国内精品久久久久久久久电影网 | 日本免费在线播放| 国内欧美视频一区二区| 欧美中文娱乐网| 91精品国产黑色瑜伽裤| 91精品国产一区二区人妖| 黑人巨大精品欧美| 激情欧美一区| 97人摸人人澡人人人超一碰| 午夜激情在线观看| 在线观看日韩高清av| 五月开心播播网| 国内久久精品| 亚洲va欧美va在线观看| aiai在线| 欧美在线观看视频一区二区| 变态另类丨国产精品| 黄色日韩在线| yellow视频在线观看一区二区 | 亚洲视频综合在线| 成人性生生活性生交12| 亚洲第一论坛sis| 韩国一区二区电影| 亚洲精品字幕在线观看| 亚洲黄色在线视频| 被黑人猛躁10次高潮视频| 日韩在线看片| 国产欧美日韩高清| 99中文字幕一区| 欧美在线免费播放| 精品人伦一区二区| 日本中文一区二区三区| 日本一区视频在线观看| 韩日成人影院| 亚洲人精品午夜在线观看| www.国产一区二区| 久久日韩精品一区二区五区| 成人综合视频在线| 亚洲综合图色| 国产精品福利网站| av资源网站在线观看| 欧美日韩一区二区三区视频| 日日操免费视频| 久久精品国产亚洲aⅴ| 亚洲精品中文字幕乱码三区不卡| 国产资源一区| 久久精品视频免费播放| 国产后入清纯学生妹| 亚洲综合图片区| 国产乱了高清露脸对白| 久久久久久久高潮| 亚洲 日韩 国产第一区| 久久久免费人体| 久久成人综合视频| 高h放荡受浪受bl| 欧美日韩国产一区二区三区| 国产男男chinese网站| 男男视频亚洲欧美| 992tv成人免费观看| 精品一区二区三区亚洲| 久久久久久国产免费| 人操人视频在线观看| 欧美亚洲日本国产| 国产女片a归国片aa| 成人精品国产免费网站| 东京热加勒比无码少妇| 日韩毛片视频| 国产精品二区三区四区| 网友自拍亚洲| 欧美床上激情在线观看| 天天色天天操天天射| 在线视频国内一区二区| 成人免费精品动漫网站| 99精品热视频| 高清av免费看| 最新亚洲激情| 亚洲欧美日韩不卡一区二区三区| 亚洲日本va午夜在线电影| 欧美综合一区第一页| 久久亚洲天堂| 亚洲精品日韩丝袜精品| 97人妻精品一区二区三区动漫| 亚洲成人动漫一区| 激情五月激情综合| av中文字幕不卡| 亚洲免费成人在线视频| 一区二区三区国产在线| 久久最新免费视频| 蜜臀av免费一区二区三区| 91香蕉电影院| 456成人影院在线观看| 久久久久久久久久久国产| 在线视频二区| 日韩高清中文字幕| 国产99对白在线播放| 欧美亚洲一区三区| 国产又色又爽又黄的| 亚洲三级视频在线观看| 色一情一交一乱一区二区三区| 国产福利一区二区三区视频在线| 亚洲精品中文字幕无码蜜桃| 欧美日韩精品免费观看视频完整| 午夜午夜精品一区二区三区文| 美女视频亚洲色图| 99高清视频有精品视频| 激情小说亚洲| 日韩免费在线免费观看| 黄色视屏在线免费观看| 欧美激情亚洲视频| 国产黄网站在线观看| 中文字幕不卡av| 日本国产在线| 精品香蕉一区二区三区| 天堂在线中文网| 欧美r级电影在线观看| 国产永久免费视频| 欧美日韩精品欧美日韩精品一综合| av黄色在线播放| 亚洲.国产.中文慕字在线| 激情五月婷婷小说| 亚洲视频一区二区在线| 天美传媒免费在线观看| 国产欧美一区二区三区鸳鸯浴| 欧美精品黑人猛交高潮| 91丨porny丨在线| 国产a级黄色片| 99久久精品免费看| 亚洲av成人片色在线观看高潮| 国产精品1区二区.| 久久久久亚洲av片无码v| 国产一区二区三区四区五区美女 | 日韩国产小视频| 欧美精品午夜| 国产精品视频网站在线观看| 欧美激情aⅴ一区二区三区| 一区二区三视频| 国产国产精品| 黑人巨茎大战欧美白妇 | 国产精品沙发午睡系列| 国产欧美一区二区三区国产幕精品| av在线观看地址| 亚洲精品激情| 人妻有码中文字幕| 人人超碰91尤物精品国产| www.激情小说.com| 精品一区二区综合| 中文字幕在线播放一区二区| 盗摄精品av一区二区三区| 日批免费观看视频| 99免费精品视频| 中国毛片在线观看| 中文字幕免费不卡在线| 最新黄色av网址| 日韩美女久久久| 久久久久久久极品内射| 欧美日韩在线视频首页| 久久黄色精品视频| 欧美午夜精品一区二区三区| 97超碰资源站| 精品国产自在久精品国产| 欧美自拍第一页| 国产亚洲欧美日韩美女| 麻豆视频网站在线观看| 欧美激情视频网址| 欧美粗大gay| 成人欧美一区二区三区在线| 99久久人爽人人添人人澡| 蜜桃传媒视频麻豆一区| 99精品小视频| 国产极品在线视频| 久久精品国产久精国产爱| 91丨porny丨九色| 久久久美女毛片| 污污的视频在线免费观看| 亚洲国产精品久久人人爱蜜臀 | 91日韩中文字幕| 欧美日韩一区二区在线| 国产精品久久久久久久免费看| 亚洲精品一区在线观看| 爱久久·www| 久久久久久久久久久人体 | 国产偷自视频区视频一区二区| 欧美特级aaa| 菠萝蜜视频在线观看一区| 丰满的亚洲女人毛茸茸| 亚洲第一综合色| 亚洲一级黄色大片| 日韩高清av一区二区三区| 国产精品一卡二卡三卡| 日本成熟性欧美| 风间由美一区二区av101| 亚洲永久一区二区三区在线| 国产精品色网| 少妇愉情理伦片bd| 中文字幕不卡在线| 国语对白永久免费| 精品奇米国产一区二区三区| 亚洲成人三级| 国产成人精彩在线视频九色| 东京久久高清| 国产日产欧美一区二区| 另类成人小视频在线| 人妻av无码一区二区三区| 午夜视频在线观看一区二区| 国产精品一区二区三区在线免费观看 | 亚洲国产网址| 高清无码视频直接看| 极品少妇xxxx精品少妇| 中文字幕在线看高清电影| 偷拍亚洲欧洲综合| 亚洲av无码一区二区乱子伦| 北条麻妃久久精品| 日本在线精品| 欧美精品国产精品久久久| 亚洲国产清纯| 美女扒开腿免费视频| 一区二区三区中文字幕| 91在线视频国产| 久久精品国产96久久久香蕉| 欧美成人福利| 亚洲一区精彩视频| 日本sm残虐另类| 在线看片中文字幕| 欧美午夜电影在线播放| 成人欧美一区| 国产精品电影在线观看| 人人狠狠综合久久亚洲婷| 天天爱天天操天天干| 国产拍揄自揄精品视频麻豆| 无码视频在线观看| 国产午夜精品全部视频在线播放| 久久sese| 视频一区二区三区免费观看| 视频一区视频二区中文字幕| 久久精品—区二区三区舞蹈| 日本高清不卡一区| аⅴ资源新版在线天堂| 国产免费一区二区三区在线观看| 日韩中文字幕高清在线观看| 天堂av在线8| 亚洲精品欧美综合四区| 成人午夜精品福利免费| 海角国产乱辈乱精品视频| 欧美a级网站| 国模杨依粉嫩蝴蝶150p| 欧美国产禁国产网站cc| 91亚洲精品国偷拍自产在线观看| 日韩视频免费在线观看| 99久久香蕉| 黄色一级大片在线观看| 国产精品乱人伦中文| 国产强伦人妻毛片| 午夜免费久久久久| 精品一区在线| 亚洲天堂av一区二区| 亚洲一区二区在线免费观看视频| 日本久久一级片| 国产福利视频一区二区| 99国产精品一区二区| 性高潮久久久久久| 欧美日韩在线视频一区| av网在线观看| 成人性色av| 久久久久久穴| 亚洲色图综合区| 日韩大陆毛片av| 欧美成a人片免费观看久久五月天| www.一区二区.com| 久久精品一区蜜桃臀影院| 91亚洲欧美激情| 57pao国产精品一区| 97精品国产| 日韩精品人妻中文字幕有码 | 午夜精品久久久久久久99黑人| 在线观看欧美理论a影院| 成人黄色一级大片| 五月婷婷综合激情| 日韩理伦片在线| 久久久久一区二区三区| 久久69国产一区二区蜜臀| www.天天色| 日韩视频永久免费观看| 亚洲精华一区二区三区| 黄色片子免费看| 91精品办公室少妇高潮对白| 欧美人与性动交α欧美精品图片| 日本10禁啪啪无遮挡免费一区二区| 国产成人av电影免费在线观看| 国产精品xxxxxx|