javascript的语言特性决定了,一定会有一个顶层对象(top object),但是根据执行环境的不同,这个顶层对象是不一样的。由执行环境决定,比如在浏览器执行环境中,顶层对象是window。而在node里,顶层对象是global
global里定义了一些全局的对象或函数,在node的任何一个模块里,都可以直接使用,比如console,setTimeout(),require()等,完整的global object document见:node.js global objects
如果想在不同的模块(文件)之间共享变量,有一个可行但是很糟糕的做法,就是借助这个global object,在global上定义的属性和函数,在任何模块里都可以访问到
global.name = "Tony";
require("./b");// 执行b.js里的代码
console.log(name);// Tony
name = "Tony";
var name = "Tony";
所以javascript的最佳实践,是强烈建议不要修改global object,只使用global上预定义的属性和函数
在模块里用var声明的变量,全部都是在module作用域里的,优先于global作用域的属性
global.name = "Tony";
require("./b");// 执行b.js里的代码
var name = "kyfxbl";
console.log(name);// kyfxbl
以上代码会输出"kyfxbl",而不是"Tony",因为module的name比global的name更优先
如果想不借助global,在不同模块之间共享代码,就需要用到exports属性。令人有些迷惑的是,在node.js里,还有另外一个属性,是module.exports。一般情况下,这2个属性的作用是一致的,但是如果对exports或者module.exports赋值的话,又会呈现出令人奇怪的结果
网上关于这个话题的讨论很多,流传最广的是这个帖子:exports vs module.exports,但是这篇帖子里有些说法明显是错误的,却没有人指出来。下面说一下我的理解
首先,exports和module.exports都是某个对象的引用(reference),初始情况下,它们指向同一个object,如果不修改module.exports的引用的话,这个object稍后会被导出
所以不管用exports还是module.exports,给这个object添加属性或函数,都是完全等效的
exports.name = "Tony";
module.exports.age = 33;
var b = require("./b");
console.log(b);// {name:"Tony", age:33}
但是有时候,希望导出的是一个构造函数,那么一般会这么写:
module.exports = function (name, age) {
this.name = name;
this.age = age;
}
exports.sex = "male";
var Person = require("./b");
var person = new Person("Tony", 33);
console.log(person);// {name:"Tony", age:33}
console.log(Person.sex);// undefined
而node导出的,永远是module.exports指向的对象,在这里就是function。所以exports指向的那个object,现在已经不会被导出了,为其增加的属性当然也就没用了
如果希望把sex属性也导出,就需要这样写:
exports = module.exports = function (name, age) {
this.name = name;
this.age = age;
}
exports.sex = "male";
所以我感觉exports根本是多余的,最终只会导出一个object,却设计了2个引用,很多时候反而会造成迷惑。exports的唯一好处就是可以少敲几个字,还不如只保留module.exports就好了
执行require之后,目标模块的代码会被完整地执行一次,然后module.exports对象被返回
需要注意的是,这个过程只会发生一次,后面重复的require,只会拿到同一个对象
exports.name = "Tony";
exports.age = 33;
var b1 = require("./b");
var b2 = require("./b");
console.log(b1 === b2); // true
console.log(b2.age);// 33
b1.age++;
console.log(b2.age);// 34
var b3=require("./b");
console.log(b3.age);// 34