好久没更新了,最近在看《深入浅出nodejs》,就摘一点点下来吧。
在CommonJS模块规范中,每个模块文件中存在着require、exports、module这3个变量,但是它们在模块文件中没有定义,是从哪里来的呢?甚至在Node的API文档中,每个模块中还有__filename、__dirname这两个变量的存在,它们又是从何而来的呢?
事实上,在编译的过程中,Node对获取的JavaScript文件内容进行了头尾包装。在头部添加了(function(exports, require, module, __filename, __dirname){\n,在尾部添加了\n}); 。一个正常的JavaScript文件会被包装成如下的样子:
(function (exports, require, module, _ _ filename, _ _ dirname) {
var math = require('math');
exports.area = function(radius){
return Math.PI * radius * radius;
};
});
这样每个模块文件之间都进行了作用域隔离。包装之后的代码会通过vm原生模块的runInThisContext()方法执行(类似eval,只是具有明确上下文,不污染全局),返回一个具体的function对象。最后,将当前模块对象的exports属性,require()方法,module(模块对象自身),以及在文件定位中得到的完整路径和文件目录作为参数传递给这个function()执行。
这就是这些变量并没有定义在每个模块文件中确存在的原因。在执行之后,模块的exports属性被返回给了调用方。exports属性上的任何方法和属性都可以被外部调用到,但是模块中的其余变量或属性则不可直接被调用。
这就是Node对CommonJS模块规范的实现。
以上摘自《深入浅出nodejs》。