@H_404_0@今天看了下node.js的require
方法的源码,终于搞清楚exports和module.exports的区别了。
@H_404_0@我们知道,node.js的模块暴露有两种方法。
@H_404_0@
1. 方式一:用exports
exports.log =function (str) {
console.log(str);
}
console.log(str);
}
var s = require("./a");
s.log("哈哈哈哈");
@H_404_0@s.log("哈哈哈哈");
2. 方式二:用module.exports
//b.js
var s = require("./a");
s("嘻嘻嘻嘻");
@H_404_0@如果将第一种方式的exports按照第二种方式写成下面这样就会出错:
var s = require("./a");
s("嘻嘻嘻嘻");
var s = require("./a");
s("哈哈哈哈");
@H_404_0@s("哈哈哈哈");
exports
和module.exports
的初始值指向的是空对象,即{}
。从源码可以看到,其实模块的require
方法实质上是调用了_load
方法,而_load
方法,最终返回的是module.exports
@H_404_0@
@H_404_0@
@H_404_0@来分析一下出错的原因。
@H_404_0@由于最开始的时候,exports
和module.exports
都指向同一个对象。
@H_404_0@第一种方式,是在给这个空对象{}
添加属性,又因为module.exports
也是指向这个对象的,所以最终require
方法返回的module.exports
是指向了这个具有log
方法的对象的,可以引用到模块。
@H_404_0@第二种方式是让module.exports
指向一片新的内存空间,exports
指向的仍然是{}
,但是由于require
方法返回的是module.exports
,所以最终也能引入模块。
@H_404_0@但是最后那种写法是让exports
指向一片新的内存空间,module.exports
指向的仍然还是{}
,那么最终require
方法是将module.exports
返回,所以会导致报错,说s不是一个function。
@H_404_0@所以:
@H_404_0@请牢记:require
方法返回的是module.exports
!
@H_404_0@以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持编程之家。 原文链接:https://www.f2er.com/nodejs/39876.html