動手實現一個AMD模塊加載器(二)

在上一篇文章中,咱們已經基本完成了模塊加載器的基本功能,接下來來完成一下路徑解析的問題。javascript

在以前的功能中,咱們全部的模塊默認只能放在同級目錄下,而在實際項目中,咱們的js頗有可能位於多個目錄,甚至是CDN中,因此如今這種路徑解析是很是不合理的,所以咱們須要將每一個模塊的name轉化爲一個絕對路徑,這樣纔是一個比較完美的解決方案。html

借鑑部分requirejs的思想,咱們能夠經過配置來配置一個baseUrl,當沒有配置這個baseUrl的時候,咱們認爲這個baseUrl就是html頁面的地址,因此咱們須要對外暴露一個config方法,以下:java

var cfg = {
  baseUrl: location.href.replace(/(\/)[^\/]+$/g, function(s, s1){
    return s1
  })
}

function config(obj) {
  obj && merge(cfg, obj);
}

function merge(obj1, obj2) {
  if(obj1 && obj2) {
    for(var key in obj2) {
      obj1[key] = obj2[key]
    }
  }
}

loadjs.config = config;複製代碼

上面的代碼中,咱們定義了一個基本的全局配置對象cfg、一個用來合併對象屬性的merge方法和一個用來支持配置的config方法。可是顯然這個時候配置baseUrl的時候須要使用一個絕對路徑。可是在實際中咱們可能更會使用的是一個相對路徑,例如../或者./或者/這個需求是很是正常的,所以咱們須要也支持這些實現。首先咱們先來寫這些的匹配的正則表達式,爲了以後的使用咱們同時也寫出檢測完整路徑(包括http、https和file協議)node

var fullPathRegExp = /^[(https?\:\/\/) | (file\:\/\/\/)]/;
  var absoPathRegExp = /^\//;
  var relaPathRegExp = /^\.\//;
  var relaPathBackRegExp = /^\.\.\//;複製代碼

同時將這些判斷寫進一個outputPath方法中。jquery

function outputPath(baseUrl, path) {
    if (relaPathRegExp.test(path)) {
      if(/\.\.\//g.test(path)) {
        var pathArr = baseUrl.split('/');
        var backPath = path.match(/\.\.\//g);
        var joinPath = path.replace(/[(^\./)|(\.\.\/)]+/g, '');
        var num = pathArr.length - backPath.length;
        return pathArr.splice(0, num).join('/').replace(/\/$/g, '') + '/' +joinPath;
      } else {
        return baseUrl.replace(/\/$/g, '') + '/' + path.replace(/[(^\./)]+/g, '');
      }
    } else if (fullPathRegExp.test(path)) {
      return path;
    } else if (absoPathRegExp.test(path)) {
      return baseUrl.replace(/\/$/g, '') + path;
    } else {
      return baseUrl.replace(/\/$/g, '') + '/' + path;
    }
  }複製代碼

這裏可能須要關注的一個相對路徑的問題,由於有多是須要返回上一級目錄的,即形如./../../的形式,所以也應該處理這種狀況。另外之因此在這裏都是要匹配baseUrl的最後一個斜槓/,是由於提供的這個頗有可能帶有斜槓,也頗有可能不帶斜槓。
最後使用config方法配置的時候,經過判斷提供的path來作相應的處理,修改config方法以下:git

function config(obj) {
    if(obj){
     if(obj.baseUrl) {
       obj.baseUrl = outputPath(cfg.baseUrl, obj.baseUrl);
     }
      merge(cfg, obj);
    } 
  }複製代碼

最後咱們修改一下每一個模塊名爲這個模塊的絕對路徑,這樣咱們就沒必要再修改loadScript方法了,咱們在loadMod方法中修改name參數,增長代碼:github

name = outputPath(cfg.baseUrl, name);複製代碼

咱們再來優化一下,畢竟若是咱們每個模塊都要使用./或者../之類的,不少模塊下這是要崩潰的,因此咱們依舊是借鑑requirejs的方法,容許使用config方法來配置path屬性這個問題,當咱們配置了一個app的path以後咱們認爲在模塊引用的時候,若是遇到app開頭則須要替換這個path。
因此先來看config方法修改以下:正則表達式

function config(obj) {
    if(obj){
     if(obj.baseUrl) {
       obj.baseUrl = outputPath(cfg.baseUrl, obj.baseUrl);
     }
     if(obj.path) {
       var base = obj.baseUrl || cfg.baseUrl;
       for(var key in obj.path) {
         obj.path[key] = outputPath(base, obj.path[key]);
       }
     }
      merge(cfg, obj);
    }
  }複製代碼

所以在loadMod方法中同時也應該檢測cfg.path中是否含有這個屬性,這時候會比較複雜,所以單獨抽出爲一個函數來講是比較好的處理方式,單獨抽出爲一個replaceName方法,以下:瀏覽器

function replaceName(name) {
    if(fullPathRegExp.test(name) || absoPathRegExp.test(name) || relaPathRegExp.test(name)) {
      return outputPath(cfg.baseUrl, name);
    } else {
      var prefix = name.split('/')[0] || name;
      if(cfg.path[prefix]) {
        if(name.split('/').length === 0) {
         return cfg.path[prefix];
        } else {
          var endPath = name.split('/').slice(1).join('/');
          return outputPath(cfg.path[prefix], endPath);
        }
      }
    }
  }複製代碼

這樣,咱們只須要在loadMod方法中調用這個方法就能夠了。
咱們再優化一下,咱們徹底能夠在define中將name替換爲一個絕對路徑,同時在主模塊加載依賴的時候,將依賴替換爲絕對路徑便可,所以咱們能夠在定義模塊的時候就將這個這個路徑替換好。
不過這個時候咱們須要明白的是,在定義模塊的時候是一個相似單詞,而聲明依賴的時候則有可能含有路徑,如何在模塊聲明的時候正確解析路徑呢?
很明顯咱們可使用一個變量來作這個事情,這個變量存儲着全部模塊名和依賴這個模塊時的聲明。那麼咱們就應該在use方法加載模塊的時候將這些變量名添加到這個變量名之下,以後再define中進行轉化,那麼最後咱們的整個代碼以下:bash

(function(root){
  var modMap = {};
  var moduleMap = {};
  var cfg = {
    baseUrl: location.href.replace(/(\/)[^\/]+$/g, function(s, s1){
      return s1
    }),
    path: {

    }
  };
  var fullPathRegExp = /^[(https?\:\/\/) | (file\:\/\/\/)]/;
  var absoPathRegExp = /^\//;
  var relaPathRegExp = /^\.\//;
  var relaPathBackRegExp = /^\.\.\//;


  function outputPath(baseUrl, path) {
    if (relaPathRegExp.test(path)) {
      if(/\.\.\//g.test(path)) {
        var pathArr = baseUrl.split('/');
        var backPath = path.match(/\.\.\//g);
        var joinPath = path.replace(/[(^\./)|(\.\.\/)]+/g, '');
        var num = pathArr.length - backPath.length;
        return pathArr.splice(0, num).join('/').replace(/\/$/g, '') + '/' +joinPath;
      } else {
        return baseUrl.replace(/\/$/g, '') + '/' + path.replace(/[(^\./)]+/g, '');
      }
    } else if (fullPathRegExp.test(path)) {
      return path;
    } else if (absoPathRegExp.test(path)) {
      return baseUrl.replace(/\/$/g, '') + path;
    } else {
      return baseUrl.replace(/\/$/g, '') + '/' + path;
    }
  }

  function replaceName(name) {
    if(fullPathRegExp.test(name) || absoPathRegExp.test(name) || relaPathRegExp.test(name) || relaPathBackRegExp.test(name)) {
      return outputPath(cfg.baseUrl, name);
    } else {
      var prefix = name.split('/')[0] || name;
      if(cfg.paths[prefix]) {
        if(name.split('/').length === 0) {
         return cfg.paths[prefix];
        } else {;
          var endPath = name.split('/').slice(1).join('/');
          return outputPath(cfg.paths[prefix], endPath);
        }
      } else {
        return outputPath(cfg.baseUrl, name);
      }
    }
  }

  function fixUrl(name) {
    return name.split('/')[name.split('/').length-1]
  }
  function config(obj) {
    if(obj){
     if(obj.baseUrl) {
       obj.baseUrl = outputPath(cfg.baseUrl, obj.baseUrl);
     }
     if(obj.paths) {
       var base = obj.baseUrl || cfg.baseUrl;
       for(var key in obj.paths) {
         obj.paths[key] = outputPath(base, obj.paths[key]);
       }
     }
      merge(cfg, obj);
    }
  }

  function merge(obj1, obj2) {
    if(obj1 && obj2) {
      for(var key in obj2) {
        obj1[key] = obj2[key]
      }
    }
  }
  function use(deps, callback) {
    if(deps.length === 0) {
      callback();
    }
    var depsLength = deps.length;
    var params = [];
    for(var i = 0; i < deps.length; i++) {
      moduleMap[fixUrl(deps[i])] = deps[i];
      deps[i] = replaceName(deps[i]);
      (function(j){
        loadMod(deps[j], function(param) {
          depsLength--;
          params[j] = param;
          if(depsLength === 0) {
            callback.apply(null, params);
          }
        })
      })(i)
    }
  }

  function loadMod(name, callback) {
    if(!modMap[name]) {
      modMap[name] = {
        status: 'loading',
        oncomplete: []
      };
      loadscript(name, function() {
        use(modMap[name].deps, function() {
          execMod(name, callback, Array.prototype.slice.call(arguments, 0));
        })
      });
    } else if(modMap[name].status === 'loading') {
      modMap[name].oncomplete.push(callback);
    } else if (!modMap[name].exports){
      use(modMap[name].deps, function() {
        execMod(name, callback, Array.prototype.slice.call(arguments, 0));
      })
    }else {
      callback(modMap[name].exports);
    }
  }

  function execMod(name, callback, params) {
    var exp = modMap[name].callback.apply(null, params);
    modMap[name].exports = exp;
    callback(exp);
    execComplete(name);
  }

  function execComplete(name) {
    for(var i = 0; i < modMap[name].oncomplete.length; i++) {
      modMap[name].oncomplete[i](modMap[name].exports);
    }
  }
  function loadscript(name, callback) {
    var doc = document;
    var node = doc.createElement('script');
    node.charset = 'utf-8';
    node.src = name + '.js';
    node.id = 'loadjs-js-' + (Math.random() * 100).toFixed(3);
    doc.body.appendChild(node);
    node.onload = function() {
      callback();
    }
  }

  function define(name, deps, callback) {
    if(moduleMap[name]) {
      name=moduleMap[name]
    } 
    name = replaceName(name);
    deps = deps.map(function(ele, i) {
      return replaceName(ele); 
    });
    modMap[name] = modMap[name] || {};
    modMap[name].deps = deps;
    modMap[name].status = 'loaded';
    modMap[name].callback = callback;
    modMap[name].oncomplete = modMap[name].oncomplete || [];
  }

  var loadjs = {
    define: define,
    use: use,
    config: config
  };

  root.define = define;
  root.loadjs = loadjs;
  root.modMap = modMap;
})(window);複製代碼

咱們進行一下測試:

loadjs.config({
      baseUrl:'./static',
      paths: {
        app: './app'
      }
    });
    loadjs.use(['app/b', 'a'], function(b) {
      console.log('main');
      console.log(b.equil(1,2));
    })複製代碼
define('a', ['app/c'], function(c) {
  console.log('a');
  console.log(c.sqrt(4));
  return {
    add: function(a, b) {
      return a + b;
    }
  }
});複製代碼
define('c', ['http://ce.sysu.edu.cn/hope/Skin/js/jquery.min.js'], function() {
  console.log('c');
  return {
    sqrt: function(a) {
      return Math.sqrt(a)
    }
  }
});複製代碼
define('b', ['c'], function(c) {
  console.log('b');
  console.log(c.sqrt(9));
  return {
    equil: function(a,b) {
      return a===b;
    }
  }
});複製代碼

打開瀏覽器咱們能夠看到正常輸出,以下:

1111
1111

說明咱們的所作的路徑解析工做是正確的。

系列文章:
動手實現一個AMD模塊加載器(一)
動手實現一個AMD模塊加載器(二)
動手實現一個AMD模塊加載器(三)

相關文章
相關標籤/搜索