编写自己的模块加载器
自己实现一个模块加载器——bodule.js
shut up, show me the code!
要想真正地了解一个加载器是如何工作的,就是自己实现一个!让我们来一步一步地实现一个名为bodule.js的模块加载器。
约定
一个模块系统,必然有一些约定,下面是bodule.js的规范。
模块
bodule.js的模块由以下几个概念组成:
- url,一个url地址对应一个模块;
- meta module:如下形式为一个meta module:
define(id, dependancies?, factory)
id必须为完整的url,dependancies如果没有依赖,则可以省略,factory包含两种形式:
Function:function(require, [exports,] [module]):
非Function:直接作为该meta模块的exports。
define('http://bodule.org/island205/venus/1.0.0/venus', ['./vango'], function (require, exports, module) {
//CommonJS
})
// or
define('http://bodule.org/island205/venus/1.0.0/conststring', 'bodule.js')
// even or
define('http://bodule.org/island205/venus/1.0.0/undefined', undefined)
dependancies中的字符串以及CommonJS中的require的参数,必须为url、相对路径或顶级路径的解析依赖于前面的id。
- 一个模块文件包含一个或多个meta module,但是,在该模块文件中,必须包含一个该模块文件url作为id的meta module,例如:
http://bodule.org/island205/venus/1.0.0/venus.js
对应的模块文件内容为:
define('http://bodule.org/island205/venus/1.0.0/venus', ['./vango'], function (require, exports, module) {
//CommonJS for venus
})
define('http://bodule.org/venus/1.0.0/vango', [], function (require, exports, module) {
//CommonJS for vango
})
该模块文件包含两个meta module,而第一个是必须的。但这两个meta模块的顺序不做要求。
简化
为了简化代码,针对
define('http://bodule.org/island205/venus/1.0.0/venus', ['./vango'], function (require, exports, module) {
//CommonJS for venus
})
这样的代码我们可以将其简化为:
define('./venus/1.0.0/venus', ['./vango'], function (require, exports, module) {
//CommonJS for venus
})
或者:
define('/venus/1.0.0/venus', ['./vango'], function (require, exports, module) {
//CommonJS for venus
})
这样的形式,然相对路径或者顶级路径必须要由一个绝对路径可参照,在bodule.js中,这个绝对路径来自于当前页面的url地址,或者使用bodule.package进行配置。
bodule cloud
在node中,可以使用require('underscore')来引用node_modules中的模块,作为bodule.js的目标,将commonjs桥接到浏览器端来使用,所以允许使用类似的写法,这种模块我们把它称作bodule模块,resovle后映射到http://bodule.org/underscore/stable
,bodule.js会在bodule.org上提供一个云服务,来支持你从这里加载这些bodule模块。
如果你想使用自己的bodule服务器,可以使用bodule.package来配置boduleServer。
npm
npm非常流行,bodule.js将其作为模块的源。我们采取与npm包一致的策略。典型的npm的package.json为(以underscore为例):
{
"name" : "underscore",
"description" : "JavaScript's functional programming helper library.",
"homepage" : "http://underscorejs.org",
"keywords" : ["util", "functional", "server", "client", "browser"],
"author" : "Jeremy Ashkenas <jeremy@documentcloud.org>",
"repository" : {"type": "git", "url": "git://github.com/jashkenas/underscore.git"},
"main" : "underscore.js",
"version" : "1.5.1",
"devDependencies": {
"phantomjs": "1.9.0-1"
},
"scripts": {
"test": "phantomjs test/vendor/runner.js test/index.html?noglobals=true"
},
"licenses": [
{
"type": "MIT",
"url": "https://raw.github.com/jashkenas/underscore/master/LICENSE"
}
],
"files" : ["underscore.js", "LICENSE"]
}
bodule.js将会使用工具将其转化为bodule模块,最终会以http://bodule.org/underscore/1.5.1
这样的地址地提供出来。注意:该地址会根据package.json中的main,变为http://bodule.org/underscore/1.5.1/underscore
。
bodule.js的API
.use
.use(id)
在页面中使用一个模块,相当于node id.js
。
.use(dependancies, factory)
在页面上定义一个即时的模块,该模块依赖于dependancies,并use该模块。等价于:
define('a-random-id', dependencies, factory)
Bodule.use('a-random-id')
.use比较简单的例子,simplest.html:
<script type="text/javascript">
Bodule.use('./a.js')
Bodule.use('/b.js')
Bodule.use(['./c.js', './d'], function (require, exports, module) {
var c = require('./c.js')
var d = require('./d')
console.log(c + d)
})
Bodule.use(['./e'], function (require) {
var e = require('./e')
console.log(e)
})
</script>
define
define(id, dependencies, factory)
定义一个meta module;
define(id, anythingNotFunction)
定义一个meta module,该模块的exports即为anythingNotFunction;
几个例子:d.js,e.js,backbone.js
.package(config)
配置模块和bodule模块的位置,还可以配置依赖的bodule模块的版本号。
Bodule.package({
cwd: 'http://bodule.org:8080/',
path: '/bodule.org/',
bodule_modules:{
cwd: 'http://bodule.org:3000/',
path: '/bower_components/',
dependencies: {
'backbone': '1.0.0'
}
}
})
完整的例子可以参考bodule.org.html。
让我们开始吧!
coffeescript
coffeescript是一门非常有趣的语言,敲起代码来很舒服,不会被JavaScript各种繁琐的细节所烦扰。所以我打算使用它来实现bodule.js。访问[coffeescript.org],上面有简洁文档,如果你熟悉JavaScript,我相信你能很快掌握CoffeeScript的。
commonjs运行时
从bodule的规范中,可以看出,它其实commonjs,或者说是commonjs wrapping的一个实现。因此,我们将直接使用commonjs的方式来组织我们的代码,你会发现,这样的代码非常清晰易读。
# This is a **private** CommonJS runtime for `bodule.js`.
# `__modules` for store private module like `util`,`path`, and so on.
modules = {}
# `__require` is used for getting module's API: `exports` property.
require = (id)->
module = modules[id]
module.exports or module.exports = use [], module.factory
# Define a module, save module in `__modules`. use `id` to refer them.
define = (id, deps, factory)->
modules[id] =
id: id
deps: deps
factory:factory
# `__use` to start a CommonJS runtime, or get a module's exports.
use = (deps, factory)->
module = {}
exports = module.exports = {}
# In factory `call`, `this` is global
factory require, exports, module
module.exports
上面这段代码是commonjs规范一种精简的表达,出自node项目中的module.js。module.js比这复杂多了,包含了多native module、读取、执行module文件、以及支持多种格式的module的事情。而我们上面这段代码就是commonjs最精简的表达,有了它,我们就可以使用common.js的方式来组织代码了。
注意,代码中的deps变量完全就是无用的,只是我觉得这样写的话,似乎更清晰一点。
define 'add', [], (require, exports, module)->
module.exports = (a, b)->
a + b
define 'addTwice', ['add'], (require, exports, module)->
add = require 'add'
exports.addTwice = (a, b)->
add add(a, b), b
use ['addTwice'], (require, exports, module)->
addTwice = require 'addTwice'
cosnole.log "#{2} + #{3} + #{3} = #{addTwice 2, 3}"
上面的代码展示了如何使用这个commonjs运行时,很简单,有木有?
很简陋?确实,我们只是用用它来组织代码,最终实现bodule.js这个复杂的commonjs运行时。
bodule API
我们改从何入手编写一个加载器呢,既然已经有了规范和接口,那我们从接口写起吧。
define 'bodule', [], (require, exports, module)->
Bodule =
use: (deps, factory)->
define: (id, deps, factory)->
package: (conf)->
module.exports = Bodule
use ['bodule'], (require, exports, module)->
Bodule = require 'bodule'
window.Bodule = Bodule
window.define = ->
Bodule.define.apply Bodule, arguments