先回答我:爲何模塊很重要?javascript
答:由於有了模塊,咱們就能夠更方便地使用別人的代碼,想要什麼功能,就加載什麼模塊。
可是,這樣作有一個前提,那就是你們必須以一樣的方式編寫模塊,不然你有你的寫法,我有個人寫法,豈不是亂了套!html
因而下面三個模塊規範出來了,這篇文章也出來了(拼出來的 {捂臉笑})。前端
JS中的模塊規範(CommonJS,AMD,CMD),若是你聽過js模塊化這個東西,那麼你就應該聽過或CommonJS或AMD甚至是CMD這些規範咯,我也聽過,但以前也真的是聽聽而已。 如今就看看吧,這些規範究竟是啥東西,幹嗎的。本文包括這三個規範的來源及對應的產物的原理。java
1、CommonJSnode
1.一開始你們都認爲JS是辣雞,沒什麼用,官方定義的API只能構建基於瀏覽器的應用程序,逗我呢,這太狹隘了吧(用了個高端詞,嘎嘎),CommonJS就按耐不住了,CommonJS API定義不少普通應用程序(主要指非瀏覽器的應用)使用的API,從而填補了這個空白。它的終極目標是提供一個相似Python,Ruby和Java標準庫。這樣的話,開發者可使用CommonJS API編寫應用程序,而後這些應用能夠運行在不一樣的JavaScript解釋器和不一樣的主機環境中。jquery
在兼容CommonJS的系統中,你可使用JavaScript開發如下程序:webpack
(1).服務器端JavaScript應用程序
(2).命令行工具
(3).圖形界面應用程序
(4).混合應用程序(如,Titanium或Adobe AIR)git
2009年,美國程序員Ryan Dahl創造了node.js項目,將javascript語言用於服務器端編程。這標誌"Javascript模塊化編程"正式誕生。由於老實說,在瀏覽器環境下,沒有模塊也不是特別大的問題,畢竟網頁程序的複雜性有限;可是在服務器端,必定要有模塊,與操做系統和其餘應用程序互動,不然根本無法編程。NodeJS是CommonJS規範的實現,webpack 也是以CommonJS的形式來書寫。程序員
node.js的模塊系統,就是參照CommonJS規範實現的。在CommonJS中,有一個全局性方法require(),用於加載模塊。假定有一個數學模塊math.js,就能夠像下面這樣加載。github
var math = require('math');
而後,就能夠調用模塊提供的方法:
var math = require('math');
math.add(2,3); // 5
CommonJS定義的模塊分爲:{模塊引用(require)} {模塊定義(exports)} {模塊標識(module)}
require()用來引入外部模塊;exports對象用於導出當前模塊的方法或變量,惟一的導出口;module對象就表明模塊自己。
雖然說Node遵循CommonJS的規範,可是相比也是作了一些取捨,填了一些新東西的。
不過,說了CommonJS也說了Node,那麼我以爲也得先了解下NPM了。NPM做爲Node的包管理器,不是爲了幫助Node解決依賴包的安裝問題嘛,那它確定也要遵循CommonJS規範啦,它遵循包規範(仍是理論)的。CommonJS WIKI講了它的歷史,還介紹了modules和packages等。
下面講講commonJS的原理以及簡易實現:
瀏覽器不兼容CommonJS的根本緣由,在於缺乏四個Node.js環境的變量。
- module
- exports
- require
- global
只要可以提供這四個變量,瀏覽器就能加載 CommonJS 模塊。
下面是一個簡單的示例。
var module = { exports: {} }; (function(module, exports) { exports.multiply = function (n) { return n * 1000 }; }(module, module.exports)) var f = module.exports.multiply; f(5) // 5000
上面代碼向一個當即執行函數提供 module 和 exports 兩個外部變量,模塊就放在這個當即執行函數裏面。模塊的輸出值放在 module.exports 之中,這樣就實現了模塊的加載。
知道了原理,就能作出工具了。Browserify 是目前最經常使用的 CommonJS 格式轉換的工具。
請看一個例子,main.js 模塊加載 foo.js 模塊。
// foo.js module.exports = function(x) { console.log(x); }; // main.js var foo = require("./foo"); foo("Hi");
使用下面的命令,就能將main.js轉爲瀏覽器可用的格式。
$ browserify main.js > compiled.js
Browserify到底作了什麼?安裝一下browser-unpack,就能看清楚了。
$ npm install browser-unpack -g
而後,將前面生成的compile.js解包。
$ browser-unpack < compiled.js [ { "id":1, "source":"module.exports = function(x) {\n console.log(x);\n};", "deps":{} }, { "id":2, "source":"var foo = require(\"./foo\");\nfoo(\"Hi\");", "deps":{"./foo":1}, "entry":true } ]
能夠看到,browerify 將全部模塊放入一個數組,id 屬性是模塊的編號,source 屬性是模塊的源碼,deps 屬性是模塊的依賴。
由於 main.js 裏面加載了 foo.js,因此 deps 屬性就指定 ./foo 對應1號模塊。執行的時候,瀏覽器遇到 require('./foo') 語句,就自動執行1號模塊的 source 屬性,並將執行後的 module.exports 屬性值輸出。
雖然 Browserify 很強大,但不能在瀏覽器裏操做,有時就很不方便。
我根據 mocha 的內部實現,作了一個純瀏覽器的 CommonJS 模塊加載器 tiny-browser-require 。徹底不須要命令行,直接放進瀏覽器便可,全部代碼只有30多行。
它的邏輯很是簡單,就是把模塊讀入數組,加載路徑就是模塊的id。
function require(p){ var path = require.resolve(p); var mod = require.modules[path]; if (!mod) throw new Error('failed to require "' + p + '"'); if (!mod.exports) { mod.exports = {}; mod.call(mod.exports, mod, mod.exports, require.relative(path)); } return mod.exports; } require.modules = {}; require.resolve = function (path){ var orig = path; var reg = path + '.js'; var index = path + '/index.js'; return require.modules[reg] && reg || require.modules[index] && index || orig; }; require.register = function (path, fn){ require.modules[path] = fn; }; require.relative = function (parent) { return function(p){ if ('.' != p.charAt(0)) return require(p); var path = parent.split('/'); var segs = p.split('/'); path.pop(); for (var i = 0; i < segs.length; i++) { var seg = segs[i];