模块在第一次加载后会被缓存,这意味着多次调用 require()
方法不会导致模块的代码被多次执行
注意:不论内置模块、用户自定义模块、还是第三方模块,他们都会优先从缓存中加载,从而提高模块的加载效率
// 自定义模块.js
console.log('ok')
require('./自定义模块.js')
require('./自定义模块.js')
require('./自定义模块.js')
内置模块的加载优先级
内置模块是由 Node.js
官方提供的模块,内置模块的加载优先级最高
例如: require('fs')
始终返回内置的 fs
模块,即使在 node_modules
目录下有名字相同的包也叫做 fs
const fs = require('fs') // 始终返回的是内置的 fs 模块
自定义模块的加载机制
-
使用 require() 加载自定义模块时,必须指定以
./
或者../
开头的路径标识符。在加载自定义模块时,如果没有指定./
或../
这样的路径标识符,则node
会把它当作内置模块
或第三方模块
进行加载 -
在使用
require()
导入自定义模块时,如果省略了文件的拓展名,则Node
会按照顺序分别尝试加载以下文件-
按照 确切的文件名 进行加载
-
补全
.js
扩展名进行加载 -
补全
.json
扩展名进行加载 -
补全
.node
扩展名进行加载 -
加载失败,终端报错
-
第三方模块的加载机制
-
如果传递给
require()
的模块标识符不是一个内置模块,也没有以'./'
或'../'
开头,则Node.js
会从当前模块的父目录开始,尝试从/node_modules
文件夹中加载第三方模块 -
如果没有找到对应的第三方模块,则移动到再上一层父目录中,进行加载,直到文件系统的根目录
-
假设在
C:\Users\itheima\project\foo.js
文件里调用了require('tools')
,则Node.js
会按以下顺序查找-
C:\Users\itheima\project\node_modules\tools
-
C:\Users\itheima\node_modules\tools
-
C:\Users\node_modules\tools
-
C:\node_modules\tools
-
目录作为模块
当把目录作为模块标识符,传递给 require()
进行加载的时候,有三种加载方式:
-
在被加载的目录下查找一个叫做
package.json
的文件,并寻找main
属性,作为require()
加载的入口 -
如果目录里没有
package.json
文件,或者main
入口不存在或无法解析,则Node.js
将会试图加载目录下的index.js
文件 -
如果以上两步都失败了,则
Node.js
会在终端打印错误消息,报告模块的缺失:Error: Cannot find module xxx