【轉】AMD、CMD、UMD模塊的寫法


原文地址:http://www.75team.com/archives/882

簡介

最近幾年,我們可以選擇的Javascript組件的生態系統一直在穩步增長。雖然陡增的選擇范圍是極好的,但當組件混合匹配使用時就會出現很尷尬的局面。開發新手們會很快發現不是所有組件都能彼此“和平相處”。

為了解決這個問題,兩種競爭關系的模塊規范AMD和CommonJS問世了,它們允許開發者遵照一種約定的沙箱化和模塊化的方式來寫代碼,這樣就能避免“污染生態系統”。

AMD

隨着RequireJS成為最流行的實現方式,異步模塊規范(AMD)在前端界已經被廣泛認同。

下面是只依賴jquery的模塊foo的代碼:

//    文件名: foo.js
define(['jquery'], function ($) {
    //    方法
    function myFunc(){};

    //    暴露公共方法
    return myFunc;
});

還有稍微復雜點的例子,下面的代碼依賴了多個組件並且暴露多個方法:

// 文件名: foo.js
define(['jquery', 'underscore'], function ($, _) {
// 方法
function a(){}; // 私有方法,因為沒有被返回(見下面)
function b(){}; // 公共方法,因為被返回了
function c(){}; // 公共方法,因為被返回了

    //    暴露公共方法
    return {
        b: b,
        c: c
    }
});


定義的第一個部分是一個依賴數組,第二個部分是回調函數,只有當依賴的組件可用時(像RequireJS這樣的腳本加載器會負責這一部分,包括找到文件路徑)回調函數才被執行。

注意,依賴組件和變量的順序是一一對應的(例如,jquery->$, underscore->_)。

同時注意,我們可以用任意的變量名來表示依賴組件。假如我們把$改成$$,在函數體里面的所有對jQuery的引用都由$變成了$$。

還要注意,最重要的是你不能在回調函數外面引用變量$和_,因為它相對其它代碼是獨立的。這正是模塊化的目的所在!

CommonJS

如果你用Node寫過東西的話,你可能會熟悉CommonJS的風格(node使用的格式與之相差無幾)。因為有Browserify,它也一直被前端界廣泛認同。

就像前面的格式一樣,下面是用CommonJS規范實現的foo模塊的寫法:

//    文件名: foo.js
//    依賴
var $ = require('jquery');
//    方法
function myFunc(){};

//    暴露公共方法(一個)
module.exports = myFunc;

還有更復雜的例子,下面的代碼依賴了多個組件並且暴露多個方法:

//    文件名: foo.js
var $ = require('jquery');
var _ = require('underscore');

//    methods
function a(){};    //    私有方法,因為它沒在module.exports中 (見下面)
function b(){};    //    公共方法,因為它在module.exports中定義了
function c(){};    //    公共方法,因為它在module.exports中定義了

//    暴露公共方法
module.exports = {
    b: b,
    c: c
};

UMD: 通用模塊規范

既然CommonJs和AMD風格一樣流行,似乎缺少一個統一的規范。所以人們產生了這樣的需求,希望有支持兩種風格的“通用”模式,於是通用模塊規范(UMD)誕生了。

不得不承認,這個模式略難看,但是它兼容了AMD和CommonJS,同時還支持老式的“全局”變量規范:

(function (root, factory) {
    if (typeof define === 'function' && define.amd) {
        // AMD
        define(['jquery'], factory);
    } else if (typeof exports === 'object') {
        // Node, CommonJS之類的
        module.exports = factory(require('jquery'));
    } else {
        // 瀏覽器全局變量(root 即 window)
        root.returnExports = factory(root.jQuery);
    }
}(this, function ($) {
    //    方法
    function myFunc(){};

    //    暴露公共方法
    return myFunc;
}));

保持跟上面例子一樣的模式,下面是更復雜的例子,它依賴了多個組件並且暴露多個方法:

(function (root, factory) {
    if (typeof define === 'function' && define.amd) {
        // AMD
        define(['jquery', 'underscore'], factory);
    } else if (typeof exports === 'object') {
        // Node, CommonJS之類的
        module.exports = factory(require('jquery'), require('underscore'));
    } else {
        // 瀏覽器全局變量(root 即 window)
        root.returnExports = factory(root.jQuery, root._);
    }
}(this, function ($, _) {
    //    方法
    function a(){};    //    私有方法,因為它沒被返回 (見下面)
    function b(){};    //    公共方法,因為被返回了
    function c(){};    //    公共方法,因為被返回了

    //    暴露公共方法
    return {
        b: b,
        c: c
    }
}));

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM