深入浅出nodejs(一) 模块加载机制

声明: 深入浅出nodejs系列文章将会在后面持续更新。

    该系列文章部分参考 朴灵《深入浅出nodejs》,并加以总结补充

你真的了解require函数吗?

看似简单的require函数, 其实内部做了大量工作。。。下面我们将详细说明require为我们所做的一切

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

文件模块加载详细流程图比较复杂, 待后续有时间更新。。。见谅

时间: 2024-08-03 14:36:47

深入浅出nodejs(一) 模块加载机制的相关文章

模块加载机制与包的加载机制

模块加载机制 包是更好的组织功能的代码结构 当前指定一个文件夹,会将文件夹内的index.js作为程序的入口    3 如果当前目录下有一个和文件夹同名的文件,优先加载文件        1 如果当前目录下有一个和文件夹同名的json文件,优先加载json  2 对于文件的加载,顺序是,如果你写上./xxx,优先会加载xxx.js/xxx.json/xxx.node 加载优先级,例如: ./abc 1:先检查当前目录下./abc目录中的package.json文件的main属性,满足就立刻加载

Angular Material串串学客户端开发 2 - Node.js模块加载机制Require()

题外话解一下博客标题,因为第一篇文章评论中,有人质疑离题很远,说了半天和Angular Material没有半毛关系.其实我的的中心在后半句<串串学客户端开发>. require() 不要把这里的Require()和RequireJS混为一谈.不过有意思的是,Typescript的模块定义,甚至同时支持这两种模块机制. 导入和使用外部模块,只是简单的一句require(),看看angular/material/docs下的编译文件gulpfile.js的代码片段.对模块导入和使用有个直观的感觉

nodejs js模块加载

nodejs的非核心模块(core module)加载主要使用的就是module.js. 项目主模块(index.js/main.js等)加载使用的应该是module.js中的runMain(),其他js模块加载流程基本上是: 1,获取js文件信息: 2,new Module(): 3,读取js文件内容,封装到一个function中,同时注入module本身,module.exports,包装过的require函数等变量: 4,在某个上下文环境中执行这个封装后的function: 5,返回mod

Node.js【6】Web开发、进阶(模块加载、控制流、部署、弊端)

笔记来自<Node.js开发指南>BYVoid编著 实现过程:https://github.com/ichenxiaodao/express-example 第5章 使用Node.js进行Web开发 从零开始用Node.js实现一个微博系统,功能包括路由控制.页面模板.数据库访问.用户注册.登录.用户会话等内容. 会介绍Express框架.MVC设计模式.ejs模板引擎以及MongoDB数据库的操作. 5.1.准备工作 Express(http://expressjs.com/)除了为http

javascript 异步模块加载 简易实现

在javascript是没有类似java或其他语言的模块概念的,因此也不可能通过import或using等关键字来引用模块,这样造成了复杂项目中前端代码混乱,变量互相影响等. 因此在复杂项目中引入AMD的概念,AMD:全称是Asynchronous Module Definition,即异步模块加载机制.通过AMD可以不需要在页面中手动添加<script>来引用脚本,而通过定义依赖自动加载模块脚本,接下来的代码将讲解如何实现建议的AMD模块,如果需要查看比较详细的实现可以下载requirejs

nodejs 模块以及加载机制,主要讨论找不到模块的问题

最主要的一个思想,加载模块无非就是找到模块在哪,只要清楚了模块的位置以及模块加载的逻辑那么找不到模块的问题就迎刃而解了.本文只是综合了自己所学的知识点进行总结,难免出现理解错误的地方,请见谅. nodejs的模块分类 1.原生模块:http  fs path等,这些模块都在源码包的lib目录下面,nodejs安装好之后是找不到这些模块的,都作为node.exe的一部分了,require这些模块永远没问题的,如果哪天出现问题了,直接重启电脑或者重装node.有什么疑问可以通过下载源码对这些原生模块

Python模块动态加载机制

本文和大家分享的主要是python中模块动态加载机制相关内容,一起来看看吧,希望对大家学习python有所帮助. import 指令 来看看 import sys 所产生的指令: co_consts : (0, None) co_names : ('sys',) 0 LOAD_CONST               0 (0) 2 LOAD_CONST               1 (None) 4 IMPORT_NAME              0 (sys) 6 STORE_NAME  

《python解释器源码剖析》第15章--python模块的动态加载机制

15.0 序 在之前的章节中,我们考察的东西都是局限在一个模块(在python中就是module)内.然而现实中,程序不可能只有一个模块,更多情况下一个程序会有多个模块,而模块之间存在着引用和交互,这些引用和交互也是程序的一个重要的组成部分.本章剖析的就是在python中,一个模块是如何加载.并引用另一个模块的功能的.对于一个模块,肯定要先从硬盘加载到内存. 15.1 import前奏曲 我们以一个简单的import为序幕 # a.py import sys 1 0 LOAD_CONST 0 (

Java类加载机制——如何实现一个工程中不同模块加载不同版本的同名jar包。

如何实现一个工程中不同模块加载不同版本的同名jar包? 曾是阿里面试的时候遇到的一个面试题.当时就有点晕,如果是平时遇到这样的问题肯定是统一加载相同版本的就好了. 而如果系统特别庞大依赖冲突繁多,涉及多部门协调,真的有必要解决这样的问题. 这个问题困扰我很久,一直没有好的解决办法.最近研究java虚拟机.终于找到了答案. 在JVM里由类名和类加载器区别不同的Java类型.因此,JVM允许我们使用不同的加载器加载相同namespace的java类,而实际上这些相同namespace的java类可以