学习文档参考于廖雪峰老师的网站
1. 模块的导出和引入:
module.exports = greet; // 导出变量名
var greet = require('./hello'); // 注意引入路径
在引入路径中,如果只写模块名,node依次会在内置模块,全局模块和当前模块下去寻找对应文件,最终可能会导致报错。这种模块加载机制被叫做CommonJs规范。在这个规范下,每个.js文件都是一个单独的模块,变量(包括函数)之间互相不影响。实现原理是因为Nodejs在运行过程中,由于支持函数链式编程性会将每个单独的.js文件用一个自运行函数包起来,因此形成闭包,而其中原本的全局变量成为了局部变量,达到每个文件中变量互不影响。
例如:
在Hello.js中,代码如下:
var s = 'Hello';
var name = 'world';
console.log(s + ' ' + name + '!');
Node.js加载了hello.js
后,它可以把代码包装一下,变成这样执行:
(function () {
// 读取的hello.js代码:
var s = 'Hello';
var name = 'world';
console.log(s + ' ' + name + '!');
// hello.js代码结束
})();
导出的原理:
Node在加载js文件之前,先准备了一个变量module
// 准备module对象:
var module = {
id: 'hello',
exports: {}
};
var load = function (module) {
// 读取的hello.js代码:
function greet(name) {
console.log('Hello, ' + name + '!');
}
module.exports = greet;
// hello.js代码结束
return module.exports;
};
var exported = load(module);
// 保存module:
save(module, exported);
而在Hello.js中使用的module实际上就是Node函数中的一个参数,通过参数把module传给load函数,hello.js就顺利把一个变量传递给了Node,而Node会把这个变量保存到某个地方,由于Node保存了所有导入的module,当我们用require()获取module时,Node找到对应的module
,把这个module
的exports
变量返回,这样,另一个模块就顺利拿到了模块的输出
module.exports与 exports的简单区别
如果要输出一个键值对象{}
,可以利用exports
这个已存在的空对象{}
,并继续在上面添加新的键值,也可以使用module.exports赋值;
如果要输出一个函数或数组,必须直接对module.exports
对象赋值。
function hello() {
console.log('Hello, world!');
}
// 方法一:
module.exports = {
hello: hello
};
// 方法二:
exports.hello = hello;
// 注意不能直接对exports以键值对方式赋值:
exports = {
hello: hello // 代码可以执行,但是模块并没有输出任何变量
}