为了让node.js的文件能够相互调用,node.js提供了一个简单的模块系统。模块是node.js应用程序基本的组成部分,文件和模块是一一对应的。换言之,一个node.js文件就是一个模块,这个文件多是javascript代码、json或者编译过的c/c++扩展。javascript
其中http、fs、net等都是node.js提供的核心模块,使用c/c++实现,外部用javascript封装。html
建立模块有两种方式,java
经过exports建立node
经过module.exports建立c++
node.js中,建立一个模块很是简单,咱们建立一个main.js文件,它引用了hello模块,代码以下,npm
var hello = require('./hello') hello.world()
在上面的代码中,require('./hello')引入了当前目录下的hello.js文件。编程
./表明当前目录,node.js默认后缀为js。json
node.js提供了exports和require两个对象,其中exports是模块公开的接口,require用于从外部获取一个模块的接口,即所获取模块的exports对象。函数
接下来咱们建立hello.js文件,以下代码所示,工具
exports.world = function() { console.log('hello world') }
以上示例中,hello.js经过exports对象把world做为模块的访问接口,在main.js中经过require('./hello')加载这个模块,而后就能够直接访问hello.js中exports对象的成员函数了。
有时候咱们只是想把一个对象封装到模块中,以下格式,
module.exports = function() { }
以上面的格式,来写一个模块,以下hello.js代码,
function Hello() { var name; this.setName = function(thyName) { name = thyName } this.sayHello = function() { console.log('hello ' + name) } } module.exports = Hello
这样就能够直接获取这个对象了,以下main.js代码,
// main.js var Hello = require('./hello') hello = new Hello() hello.setName('BYVoid') hello.sayHello()
模块接口的惟一变化是使用module.exports = Hello代替了exports.world = function() {}。在外部引用该模块时,其接口对象就是要输出的Hello对象自己,而不是原先的exports。
为了更好地解释exports和module.exports之间的关系,先经过一个简单的js示例来作一个说明,以下代码,
var a = {name: 1} var b = a console.log(a) console.log(b) b.name = 2 console.log(a) console.log(b) b = {name: 3} console.log(a) console.log(b)
运行test.js结果为,
{ name: 1 } { name: 1 } { name: 2 } { name: 2 } { name: 2 } { name: 3 }
简单解释一下上面的代码:a是一个对象,b是对a对象的引用,此时a和b只想同一块内存,因此前两个输出同样;当对b作修改时,则a和b只想同一块内存地址的内容发生了改变,因此a的值改变也体现了出来;当b被覆盖时,b只想了一块新的内存,而a仍是只想原来的内存,因此最后两个输出不同。
明白了上面的例子以后,只须要指点3点就能了解exports和module.exports的区别了,
module.exports初始值为一个空对象{}
exports是只想module.exports的引用
require()返回的是module.exports而不是exports
也就是说,module.exports才是真正的接口,exports只不过是它的一个辅助工具。最攻返回给调用者的是module.exports而不是exports。
再强调一点,在node.js中,一个文件对应一个模块。为了方便,模块中会有一个exports对象,它和module.exports指向同一个变量,因此咱们修改exports对象的时候也会修改module.exports对象;当咱们经过赋值方式为module.exports赋值时候,此时module.exports与exports对象指向的变量就不一样了,因此不管exports对象怎么改,都和module.exports对象没有任何关系了。
加粗!加粗!加粗!通常来讲,推荐使用module.exports,尽可能少使用exports。
在node.js中模块有两种类型,即,
核心模块
文件模块
核心模块直接使用名称获取,例如常常使用的http模块,使用以下代码获取,
var http = require('http') ... http.createServer()
简要描述一下上面的代码,node.js中自带了一个叫作http的模块,在上述代码中咱们请求它并把返回的值赋值给一个本地变量,这样本地变量就编程了一个拥有全部http模块所提供的公共方法的对象。
在前面建立模块的demo中,经过require('./hello')语法,以下代码,
var Hello = require('./hello') hello = new Hello() hello.setName('BYVoid') ... ...
这里,咱们使用./test来获取自定义文件模块,这种经过相对路径或绝对路径是文件模块的搜索方式。
node.js加载模块时,遵循了以下的加载规则,
核心模块优先级最高,直接使用名字加载,再有命名冲突的时候首先加载核心模块
文件模块只能按照路径加载 -- 相对路径或绝对路径,而且能够省略默认的.js后缀名
查找node_modules目录,当咱们在调用npm install <name>命令的时候,会在当前目录下建立node_module目录来安装模块,当require遇到一个既不是核心模块,又不是以路径形式表示的模块名称时,会试图在当前目录下的node_modules目录中查找是否是有这样一个模块。若是没有找到,则会在当前目录的上一层的node_modules目录中继续查找,反复执行这一过程,知道遇到根目录位置。
相对路径 - 例如:
./hello
表示同级目录,../hello
表示上层目录绝对路径 - 例如:
/Users/user/Desktop/js/hello