前端模塊化方案學(xué)習(xí)

我在目前的項(xiàng)目中遇到了一個匪夷所思的問題:CommonJS規(guī)范是同步加載模塊資源,為什么前端項(xiàng)目中webpack使用的寫法卻可以遵循這個偏服務(wù)器端的一個規(guī)范呢?為什么不用適合瀏覽器異步加載策略的AMD規(guī)范呢?

首先要理解一下CommonJS、AMD兩種規(guī)范各是什么意思?

CommonJS的核心思想是把一個文件當(dāng)做一個模塊,要在哪里使用這個模塊,就在哪里require這個模塊,然后require方法開始加載這個模塊并且執(zhí)行其中的代碼,最后會返回你指定的export對象。例如:

a.js中

    module.export = function() {
        hello: function() {
            alert("你好");
        }
    }

你在其它js中可以這樣

var a = require('./xxx/a.js');
a.hello(); // ==> 彈窗“你好”

順便在這里說一下commonJS中有的地方用module.export = xxx,有的地方用export = xxx,其實(shí)它們沒有多大區(qū)別,node.js中 module是個對象,export是其中的一個屬性,而export就是對moudle.export這個屬性的一個引用罷了。
參考鏈接:node.js中module.export與export的區(qū)別。

BUT
CommonJS 加載模塊是同步的,所以只有加載完成才能執(zhí)行后面的操作,不能非阻塞的并行加載多個模塊。像Node.js主要用于服務(wù)器的編程,加載的模塊文件一般都已經(jīng)存在本地硬盤,所以加載起來比較快,不用考慮異步加載的方式,所以CommonJS規(guī)范比較適用。但如果是瀏覽器環(huán)境,要從服務(wù)器加載模塊,這是就必須采用異步模式。所以就有了 AMD CMD 解決方案。

AMD和CMD規(guī)范都是異步加載的方式,CMD沒有具體了解過就不多說了,它的典型實(shí)現(xiàn)是SeaJS,據(jù)說是按需加載就近原則,as lazy as posible.

AMD全稱(Asynchromous Module Definition),它主要是采用了異步加載模塊的方式,可以并行加載多個模塊,等所有模塊都加載并且解釋執(zhí)行完成后,才會執(zhí)行接下來的代碼(包括用AMD規(guī)范轉(zhuǎn)換成CommonJS模塊定義),講究一個“提前執(zhí)行”的思想,RequireJS就是對AMD規(guī)范最直接的實(shí)現(xiàn)。

下面用RequireJS來執(zhí)行幾個模塊,看看會發(fā)生什么。
a.js

define(function() {
    console.log("I am module a, I have been required and excuted");
});

b.js

define(['c'], function(c) {
    console.log("I am module b, I have been required and excuted, I depend on module c");
});

c.js

define(function() {
    console.log("I am module c, I have been required and excuted");
});

main.js

//轉(zhuǎn)換成CommonJS模塊定義
define(function(require, exports, module) {
    require("a");
    console.log("a required");

    require("b");
    console.log("b required");

    console.log("all modules have been required");
});

然后你覺得結(jié)果會是這樣嗎?

QQ圖片20170228200910.png

No! No! No!
如果你用的是實(shí)現(xiàn)CommonJS規(guī)范的Browserify或者用webpack來運(yùn)行這些模塊的話,結(jié)果確實(shí)是上述那樣,上圖就是CommonJS規(guī)范所得到的結(jié)果。

實(shí)際結(jié)果是:

QQ截圖20170228201817.png

仔細(xì)一想,RequireJS是標(biāo)準(zhǔn)的采用AMD規(guī)范異步加載方式的,也就是說不管你用了AMD規(guī)范轉(zhuǎn)換成CommonJS規(guī)范寫的模塊(偽同步,和CommonJS寫法相同而已實(shí)際還是異步)還是直接異步加載,AMD規(guī)范始終遵循一切模塊皆優(yōu)先加載并執(zhí)行,所以就算把require模塊寫到某句執(zhí)行的代碼后面,它仍然會被拉到最先去執(zhí)行。

這里想記一下我查詢資料得知的一些關(guān)于模塊加載和解析執(zhí)行的東西。

一個模塊被require之后它會經(jīng)歷兩個步驟,首先是加載,這個加載,可以是從本地加載的,比如webpack打包后的模塊,也可以是從服務(wù)器請求來的模塊資源。接著模塊中的代碼會被瀏覽器解釋執(zhí)行。CommonJS規(guī)范和AMD規(guī)范最大區(qū)別就體現(xiàn)在加載那一步驟上,如果都是從本地加載的,ok,是不是并行加載無所謂,反正都很快,如果是前端請求服務(wù)器獲取模塊資源,CommonJS的同步加載方式就坑爹了,它會由于某個請求加載的時間過長導(dǎo)致瀏覽器阻塞,不會往下執(zhí)行,所以就會出現(xiàn)網(wǎng)頁打開緩慢的現(xiàn)象。但是CommonJS和AMD的模塊執(zhí)行那一步所用的時間是一樣的。

所以這就很好解釋了為什么webpack可以讓前端模塊開發(fā)使用CommonJS,原因是無所謂。webpack也支持AMD、CMD規(guī)范,我自己試驗(yàn)過用無論是采用CommonJS還是AMD規(guī)范編寫模塊,最后都會被webpack分析依賴關(guān)系,然后打包到本地。之后瀏覽器加載的時候其實(shí)都是在讀取本地文件(我的理解,總之同步異步?jīng)]有什么區(qū)別了)。

最后放出一張我覺得很好地幫我理解RequireJS、SeaJS和Browserify、Webpack的圖

Paste_Image.png

RequireJS和SeaJS都是在線編譯的,就是在瀏覽器上加了一個CommonJS和AMD規(guī)范的解釋器解釋執(zhí)行。
browserify和webpack是預(yù)編譯,在本地將你寫的CommonJS和AMD模塊編譯裝換成瀏覽器認(rèn)識的JS。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 寫在開頭 先說說為什么要寫這篇文章, 最初的原因是組里的小朋友們看了webpack文檔后, 表情都是這樣的: (摘...
    Lefter閱讀 5,432評論 4 31
  • 1.幾種基本數(shù)據(jù)類型?復(fù)雜數(shù)據(jù)類型?值類型和引用數(shù)據(jù)類型?堆棧數(shù)據(jù)結(jié)構(gòu)? 基本數(shù)據(jù)類型:Undefined、Nul...
    極樂君閱讀 5,869評論 0 106
  • 原文鏈接:http://www.cnblogs.com/lvdabao/p/js-modules-develop....
    舌尖上的大胖閱讀 839評論 0 1
  • 當(dāng)一個點(diǎn)擊事件產(chǎn)生后,它的傳遞過程:Activity->ViewGroup->View。頂級View接收到事件后,...
    PeOS閱讀 352評論 0 0
  • 轉(zhuǎn)眼三個月已過,前方戰(zhàn)事雖吃緊,但是捷報(bào)頻傳,荊家軍果然不同凡響。只是我的心里總有那么一絲忐忑,所喜九衡已經(jīng)歸來,...
    花落寂寂hjr閱讀 351評論 3 5

友情鏈接更多精彩內(nèi)容