日期:2014-05-16  浏览次数:20392 次

node.js的global variable,和module.exports

global


javascript的语言特性决定了,一定会有一个顶层对象(top object),但是根据执行环境的不同,这个顶层对象是不一样的。由执行环境决定,比如在浏览器执行环境中,顶层对象是window。而在node里,顶层对象是global

global里定义了一些全局的对象或函数,在node的任何一个模块里,都可以直接使用,比如console,setTimeout(),require()等,完整的global object document见:node.js global objects

如果想在不同的模块(文件)之间共享变量,有一个可行但是很糟糕的做法,就是借助这个global object,在global上定义的属性和函数,在任何模块里都可以访问到

global.name = "Tony";

然后在a.js里

require("./b");// 执行b.js里的代码

console.log(name);// Tony

另外一种更简单(也更容易引起错误)的做法,是不用var关键字声明变量,也会成为global的属性

name = "Tony";

但是,如果加上了var关键字,声明的变量就只局限在module(当前文件)的作用域中,所以强烈推荐显式地用var来声明变量,以免不小心挂到了global上

var name = "Tony";

为什么依赖global是不好的实践呢?因为所有的模块都可以不受限制地使用global,而且缺少命名空间的约束,非常容易引起冲突,从而引发潜在的BUG。而且这种BUG一旦发生,要定位是极其困难的,不知道是在哪里改变了全局变量而引发的问题

所以javascript的最佳实践,是强烈建议不要修改global object,只使用global上预定义的属性和函数


module


在模块里用var声明的变量,全部都是在module作用域里的,优先于global作用域的属性

global.name = "Tony";

require("./b");// 执行b.js里的代码

var name = "kyfxbl";

console.log(name);// kyfxbl

以上代码会输出"kyfxbl",而不是"Tony",因为module的name比global的name更优先


module.exports vs exports


如果想不借助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}

所以如果只是给对象添加属性,不改变exports和module.exports的引用目标的话,是完全没有问题的

但是有时候,希望导出的是一个构造函数,那么一般会这么写:

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

这个sex属性是不会导出的,因为引用关系已经改变:


而node导出的,永远是module.exports指向的对象,在这里就是function。所以exports指向的那个object,现在已经不会被导出了,为其增加的属性当然也就没用了

如果希望把sex属性也导出,就需要这样写:

exports = module.exports = function (name, age) {
    this.name = name;
    this.age = age;
}

exports.sex = "male";

事实上,查看很多module的源代码,会发现就是这么写的,这时的引用关系:


所以我感觉exports根本是多余的,最终只会导出一个object,却设计了2个引用,很多时候反而会造成迷惑。exports的唯一好处就是可以少敲几个字,还不如只保留module.exports就好了


exports的缓存


执行require之后,目标模块的代码会被完整地执行一次,然后module.exports对象被返回

需要注意的是,这个过程只会发生一次,后面重复的require,只会拿到同一个对象

exports.name = "Tony";
exports.age = 33;

var