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

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

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

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

a.js中

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

你在其它js中可以這樣

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

順便在這里說(shuō)一下commonJS中有的地方用module.export = xxx羹令,有的地方用export = xxx,其實(shí)它們沒(méi)有多大區(qū)別损痰,node.js中 module是個(gè)對(duì)象福侈,export是其中的一個(gè)屬性,而export就是對(duì)moudle.export這個(gè)屬性的一個(gè)引用罷了卢未。
參考鏈接:node.js中module.export與export的區(qū)別肪凛。

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

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

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

下面用RequireJS來(lái)執(zhí)行幾個(gè)模塊耳胎,看看會(huì)發(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");
});

然后你覺(jué)得結(jié)果會(huì)是這樣嗎惕它?

QQ圖片20170228200910.png

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

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

QQ截圖20170228201817.png

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

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

一個(gè)模塊被require之后它會(huì)經(jīng)歷兩個(gè)步驟劝术,首先是加載,這個(gè)加載呆奕,可以是從本地加載的养晋,比如webpack打包后的模塊,也可以是從服務(wù)器請(qǐng)求來(lái)的模塊資源梁钾。接著模塊中的代碼會(huì)被瀏覽器解釋執(zhí)行绳泉。CommonJS規(guī)范和AMD規(guī)范最大區(qū)別就體現(xiàn)在加載那一步驟上,如果都是從本地加載的姆泻,ok零酪,是不是并行加載無(wú)所謂,反正都很快拇勃,如果是前端請(qǐng)求服務(wù)器獲取模塊資源四苇,CommonJS的同步加載方式就坑爹了,它會(huì)由于某個(gè)請(qǐng)求加載的時(shí)間過(guò)長(zhǎng)導(dǎo)致瀏覽器阻塞方咆,不會(huì)往下執(zhí)行月腋,所以就會(huì)出現(xiàn)網(wǎng)頁(yè)打開緩慢的現(xiàn)象。但是CommonJS和AMD的模塊執(zhí)行那一步所用的時(shí)間是一樣的瓣赂。

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

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

Paste_Image.png

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

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末颓屑,一起剝皮案震驚了整個(gè)濱河市斤寂,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌揪惦,老刑警劉巖遍搞,帶你破解...
    沈念sama閱讀 206,214評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異器腋,居然都是意外死亡溪猿,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,307評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門纫塌,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)诊县,“玉大人,你說(shuō)我怎么就攤上這事措左∫廊” “怎么了?”我有些...
    開封第一講書人閱讀 152,543評(píng)論 0 341
  • 文/不壞的土叔 我叫張陵怎披,是天一觀的道長(zhǎng)抗悍。 經(jīng)常有香客問(wèn)我,道長(zhǎng)钳枕,這世上最難降的妖魔是什么缴渊? 我笑而不...
    開封第一講書人閱讀 55,221評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮鱼炒,結(jié)果婚禮上衔沼,老公的妹妹穿的比我還像新娘。我一直安慰自己昔瞧,他們只是感情好指蚁,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,224評(píng)論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著自晰,像睡著了一般凝化。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上酬荞,一...
    開封第一講書人閱讀 49,007評(píng)論 1 284
  • 那天搓劫,我揣著相機(jī)與錄音,去河邊找鬼混巧。 笑死枪向,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的咧党。 我是一名探鬼主播秘蛔,決...
    沈念sama閱讀 38,313評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了深员?” 一聲冷哼從身側(cè)響起负蠕,我...
    開封第一講書人閱讀 36,956評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎倦畅,沒(méi)想到半個(gè)月后虐急,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,441評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡滔迈,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,925評(píng)論 2 323
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了被辑。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片燎悍。...
    茶點(diǎn)故事閱讀 38,018評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖盼理,靈堂內(nèi)的尸體忽然破棺而出谈山,到底是詐尸還是另有隱情,我是刑警寧澤宏怔,帶...
    沈念sama閱讀 33,685評(píng)論 4 322
  • 正文 年R本政府宣布奏路,位于F島的核電站,受9級(jí)特大地震影響臊诊,放射性物質(zhì)發(fā)生泄漏鸽粉。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,234評(píng)論 3 307
  • 文/蒙蒙 一抓艳、第九天 我趴在偏房一處隱蔽的房頂上張望触机。 院中可真熱鬧,春花似錦玷或、人聲如沸儡首。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,240評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)蔬胯。三九已至,卻和暖如春位他,著一層夾襖步出監(jiān)牢的瞬間氛濒,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,464評(píng)論 1 261
  • 我被黑心中介騙來(lái)泰國(guó)打工鹅髓, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留泼橘,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,467評(píng)論 2 352
  • 正文 我出身青樓迈勋,卻偏偏與公主長(zhǎng)得像炬灭,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,762評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容

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