node.js中使用CommonJS规范实现模块功能,一个单独的文件就是一个单独的模块。通过require方法实现模块间的依赖管理。
通过require加载模块,是同步操作。
加载流程如下:
1、找到需要加载的模块文件。
2、判断是否缓存过,如果没有,则读取模块文件的内容。
3、把读取到的内容,封装在一个函数里运行。
(function (exports, require, module, __filename, __dirname) { //模块的代码实际上在这里 });
4、返回 module.exports 需要导出的内容。
exports 表示模块的导出对象
require 表示require方法
module 表示当前模块对象
__filename 表示当前文件的绝对路径
__dirname 表示当前文件夹的绝对路径
在模块的内部,this 指向的是 当前模块的导出对象
console.log(this === module.exports); console.log(this === exports);
module对象有不少属性,可以打印出来看看。
console.log(module);
module.id 表示模块的标识符。
module.exports 表示模块的导出对象。
module.parent 表示当前模块的父模块,当前模块是谁加载的。
module.filename 表示模块的绝对路径。
module.loaded 表示是否加载完成。
module.children 表示当前模块加载了哪些模块。
module.paths 表示模块的搜索路径,路径的多少取决于目录的深度。
node.js通过require方法加载文件是同步的,为了提高效率,会缓存已经加载过的文件。
console.log(require.cache);
缓存的key是模块的绝对路径,value是module对象。
require.resolve获取模块的绝对路径,但并不会加载模块。
console.log(require.resolve('./01.js'));
require.main获取入口模块
console.log(require.main);
require.extensions获取模块支持的扩展名
node.js中模块文件分三种:1、.js模块 2、.json文件模块 3、.node C++扩展二进制模块
console.log(require.extensions);
node.js通过require加载文件时,比如 require('./user') 会先找 user 文件,然后没找到,再找 user.js,再找 user.json,再找 user.node。
node.js的模块分类:
1、核心模块,编译二进制,加载速度最快,像 fs,http,events等。
2、文件模块,保存在硬盘上,加载速度比较慢,通过名称或路径来加载。
3、第三方模块,只指定名称则从node_modules目录下加载,查询的路径包括module.paths和全局目录。
全局目录:
windows中:环境变量NODE_PATH指定的路径。
linux中:$HOME/.node_modules和$HOME/.node_libraries目录
文件模块查找的规则,比如 require('./user') :
1、判断加载的文件是否以绝对路径加载的。
2、如果是绝对路径,则直接加载绝对路径的文件。
3、如果不是,判断文件有没有以 './' ,'../' 开头,如果有则找到该相对路径,如果没有则通过modules.paths和全局目录获取路径。
4、尝试在第3步中的路径下查找 user 文件。
5、如果没有找到,则尝试加扩展名( user.js,user.json,user.node)进行查找。
6、如果还没有,尝试根据包进行查找,user 目录下,包文件描述 package.json 中 main 配置的文件进行加载。
7、如果没有 package.json,则查找 user 目录下, index.js 和 index.node。
8、如果还没有,则找下一个modules.paths。
9、最后还是没有找到,则文件加载失败。