nodejs之模块加载机制

 

nodejs模块加载原理

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));
};
相关文章
相关标签/搜索