Node编译过程
在Node的每个文件模块中都存在这require,exports,module这几个变量,我们可以直接使用,但是看不到它的声明?为什么?
事实上,Node对获取来的JavaScript进行了头尾包装,一个正常的模块会包装成这样:
// gulpfile.js
var gulp = require('gulp');
gulp.task('help', function() {
console.log('----------------------------------------');
console.log('gulp 开发模式');
console.log('gulp build 部署模式');
console.log('----------------------------------------');
});
=>
(function (exports, require, module, __filename, __dirname) {
var gulp = require('gulp');
gulp.task('help', function() {
console.log('-----------------------------------');
console.log('gulp 开发模式');
console.log('gulp build 部署模式');
console.log('-----------------------------------');
});
});
经过Node的头尾包装之后,我们就能使用require,exports,module,__filename等变量了,因为这些变量都被当成参数传递过来。(JavaScript中没有引用传参)
模块对象
在Node中,每个文件模块都是一个对象,它的定义如下:
function Module(id, parent) {
this.id = id;
this.exports = {};
this.parent = parent;
if(parent && parent.children) {
parent.children.push(this);
}
this.filename = null;
this.loaded = false;
this.children = [];
}
头尾包装阶段传入的是module是Module类的一个实例对象。而第一个参数exports实际上是module.exports的一个别名。
var exports = module.exports = {};
引入模块
CommonJS规范中使用require来引入模块,在Node中的实现你可以简化成为这样(实际上要复杂得多!):
var require = function(path) {
// 做爱做的事
return module.exports;
};
举个栗子,假设我们定义了A模块
// A.js
exports.sayHello = function() {
console.log('Hello');
}
我们在其他地方引用:
var a = require('./A');
a.sayHello();
按照上面的简化版实现,引用部分代码会变成这样:
var a = {
sayHello: function() {
console.log('Hello');
}
}
a.sayHello();
看到这里,相信客官你肯定能理解了(恍然大悟 焕然一新 恍恍惚惚)。所以exports只是module.exports的别名,也就是一个指针,指向module.exports对象所在的地址。如果你直接将一个对象赋给exports,那么改变的是exports指向的值,而module.exports对象没有改变。
用过express的话,那你应该也看过这样的写法。
var app = require('express')();
其实就是把一个函数赋值给module.exports对象。
module.exports = function() {
}
如果你写成exports = function,那就会报错。
多说两句
以上内容为个人学习总结,新手入门,如理解有误,请一定要告诉我~
最近才开始接触Node,第一印象还不错,比PHP框架轻太多。学完以后就能靠JavaScript完成一统天下的大业。哈哈哈cd(虽然一直都觉得JavaScript是一门随便的语言)
《深入浅出Node.js》绝对是一本好书,但是不适合我入门,额。
2016开年第一炮打完了。
如需转载,请注明出处:http://w3ctrain.com/2016/01/01/understanding-module-exports-exports-node/