nodejs模块加载原理

node加载模块步骤:

  1) 路径分析 (如判断是不是核心模块、是绝对路径还是相对路径等)

  2) 文件定位 (文件扩展名分析, 目录和包处理等细节)

  3) 编译执行

原生模块加载顺序

  1) 缓存

  2) 本地原生模块

文件模块加载顺序

  1) 缓存

  2) 如果是绝对路径, 则直接按路径读取并编译

  3) 如果是“/”则直接从/node_modules目录查找

  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));
};

nodejs之模块加载机制的更多相关文章

  1. nodejs(13)模块加载机制

    模块加载机制 优先从缓存中加载 当一个模块初次被 require 的时候,会执行模块中的代码,当第二次加载相同模块的时候,会优先从缓存中查找,看有没有这样的一个模块! 好处:提高模块的加载速度:不需要 ...

  2. 【前端】CommonJS的模块加载机制

    CommonJS的模块加载机制 CommonJS模块的加载机制是,输入的是被输出的值的拷贝.也就是说,一旦输出一个值,模块内部的变化就影响不到这个值. 例如: // lib.js var counte ...

  3. Dojo初探之1:AMD规范,编写符合AMD规范(异步模块加载机制)的模块化JS(其中dojo采用1.11.2版本)

    一.AMD规范探索 1.AMD规范(即异步模块加载机制) 我们在接触js的时候,一般都是通过各种function来定义一些方法,让它们帮我们做一些事情,一个js可以包含很多个js,而这些functio ...

  4. Skywalking-13:Skywalking模块加载机制

    模块加载机制 基本概述 Module 是 Skywalking 在 OAP 提供的一种管理功能特性的机制.通过 Module 机制,可以方便的定义模块,并且可以提供多种实现,在配置文件中任意选择实现. ...

  5. nodejs js模块加载

    本文地址:http://www.cnblogs.com/jasonxuli/p/4381747.html nodejs的非核心模块(core module)加载主要使用的就是module.js. 项目 ...

  6. Node.js中模块加载机制

    1.模块查找规则-当模块拥有路径但没有后缀时:(require(‘./find’)) require方法根据模块路径查找模块,如果是完整路径,直接引入模块: 如果模块后缀省略,先找同名JS文件,再找同 ...

  7. node模块加载机制。

  8. javascript 异步模块加载 简易实现

    在javascript是没有类似java或其他语言的模块概念的,因此也不可能通过import或using等关键字来引用模块,这样造成了复杂项目中前端代码混乱,变量互相影响等. 因此在复杂项目中引入AM ...

  9. browserify.js 的模块加载

    browserify的模块加载基本上和nodejs的类似: nodejs 的模块加载是依次去读取文件然后用一个类eval() 函数执行并返回module.exports的结果.为了避免循环加载,在加载 ...

随机推荐

  1. mysql 执行多线程临时方案

    sqr::IDatabase *db=NULL;IDbConnection *conn = NULL;int main(int argc, char* argv[]) { db = GetDataba ...

  2. const变量指针赋值给非const类型的指针运行结果

    在c++可以定义一个const变量,然后把变量的值赋给一个非const指针,可以通过指针来改变const变量的值吗?下面的截图给出了答案

  3. MyEclipse配置Maven插件

    一.工具环境 1.jdk-7u80-windows-x64 2.apache-tomcat-7.0.70 3.apache-maven-3.3.9 4.MyEclipse 10.7 5.windows ...

  4. 2018.10.16 NOIP模拟 膜法(组合数学)

    传送门 原题,原题,全TM原题. 不得不说天天考原题. 其实这题我上个月做过类似的啊,加上dzyodzyodzyo之前有讲过考试直接切了. 要求的其实就是∑i=lr(ii−l+k)\sum _{i=l ...

  5. 2018.09.21 atcoder An Invisible Hand(贪心)

    传送门 简单贪心啊. 这题显然跟t并没有关系,取差量最大的几组买入卖出就行了. 于是我们统计一下有几组差量是最大的就行了. 代码: #include<bits/stdc++.h> #def ...

  6. 【C#】VS2015开发环境的安装和配置(转)

    出处: http://www.cnblogs.com/rainmj/p/5636518.html http://www.cnblogs.com/rainmj/p/5636529.html http:/ ...

  7. hadoop学习笔记(六):HBase体系结构和数据模型

    1. HBase体系结构 一个完整分布式的HBase的组成示意图如下,后面我们再详细谈其工作原理. 1)Client 包含访问HBase的接口并维护cache来加快对HBase的访问. 2)Zooke ...

  8. Netty Nio启动全流程

    Netty Nio启动全流程 1. 各组件之间的关系 说明:EventLoopGroup类似线程池,EventLoop为单线程,每个EventLoop关联一个Nio Selector,用于注册Chan ...

  9. 从hbase到hive,以及sqoop转到mysql解析

    https://blog.csdn.net/qq_33689414/article/details/80328665 hive关联hbase的配置文件 hive和hbase同步https://cwik ...

  10. shell 脚本 删除文件内容为空的文件

    #!/bin/bask # cd /tmp for a in * ;do if [ ! -s $a ] ;then #[ ! -s $a ] 文件为空返回为真 rm -rf $a fi done 测试 ...