node加载模块步骤:node
1) 路径分析 (如判断是否是核心模块、是绝对路径仍是相对路径等)c++
2) 文件定位 (文件扩展名分析, 目录和包处理等细节)json
3) 编译执行 数组
原生模块加载顺序缓存
1) 缓存app
2) 本地原生模块函数
1) 缓存ui
2) 若是是绝对路径, 则直接按路径读取并编译spa
3) 若是是“/”则直接从/node_modules目录查找code
4) 若是是相对路径, 则生成以下查询规则,
[ '/home/myapp/mydir/node_module', '/home/myapp/node_module' '/home/node_module', '/node_module' ]
5) 从上述数组中取出第一个目录做为查找对象, 若是存在结束查找
6) 而后依次尝试添加.js、.json、.node后缀继续查找, 若是存在则结束
7) 尝试将require参数做为一个包查找, 读取目录下的package.json文件, 取得main参数指定的文件
8) 根据指定的文件未找到, 若是没有,执行第6步
9) 若是main参数不存在或者第8步未找到, 则查找该目录下index文件, 若是没有, 执行第6步
10) 若是依然没有找到, 则开始取出数组第二条路径, 而后执行5-7步。 直到数组中最后一个值
11) 若是还没找到, 抛出异常
至此, 文件终于找到了。。。而后呢?找到后该作什么呢?
也许有的人早就发现一个问题, require函数是哪里来的呢?模块中明明没有定义啊, 为何就能使用了呢?
有的同窗立刻回答说, 它是个全局的。。。全局的?那这个所谓全局的又在哪定义的呢? 额。。。不知道。。。
require到底哪里来的呢?
上面咱们已经通过重重困难终于找到了咱们的文件, 下一步就是咱们的编译
node针对不一样后缀的文件分类编译
1) .js文件的编译
.js文件的编译源码比较复杂, 其最终编译后会包装成以下结构
(function (exports, require, module, __filename, __dirname) { var math = require('math'); exports.area = function(radius) { return Math.PI * radius * radius; } })
如今知道为何有require, exports, module这些函数或对象了吧。。。
2) .json文件的编译
.json的文件最为简单, 其实就是调用JSON.parse。下为node源码
Module._extensions['.json'] = function(module, filename) { var content = fs.readFileSync(filename, 'utf8'); try { module.exports = JSON.parse(internalModule.stripBOM(content)); } catch (err) { err.message = filename + ': ' + err.message; throw err; } };
3) .node文件的编译
.node是c/c++模块, 在此不深究, 附上源码
Module._extensions['.node'] = function(module, filename) { return process.dlopen(module, path._makeLong(filename)); };