Nodejs(模块机制)

在Node 中引入模块,需要经历三个步骤

  1. 路径分析
  2. 文件定位
  3. 编译执行
    在Node中,模块分为两类:一类是Node提供的模块,称为核心模块;一类是用户编写的模块,称为文件模块。
  • 核心模块部分在Node源码的编译过程中,编译进了二进制执行文件。在Node进程启动时,部分核心模块就被直接加载进内存中,所以这部分核心模块引入时,文件定位和编译执行两个步骤可以省略掉,并且在路径分析中优先判断,所以它的加载速度是最快的。

  • 文件模块则是在运行时动态加载,需要完整的路径分析、文件定位、编译执行过程,速度比核心模块慢。

优先从缓存加载

无论是核心模块还是文件模块,require() 方法对相同模块的二次加载都一律采用缓存优先的方式,不同之处在于核心模块的缓存检查优先于文件模块的缓存检查。

路径分析和文件定位

模块标识符分析

require()方法接受一个标识符作为参数。标识符在Node中主要分为一下几类:

  • 核心模块,例如http、fs、path等

  • .或..开始的相对路径文件模块

  • 以/开始的绝对路径文件模块

  • 非路径形式的文件模块,例如自定义模块

核心模块

核心模块加载优先级仅次于缓存,它在Node源码编译过程中已经编译为二进制代码,其加载过程最快

路径形式的文件模块

在分析路径模块时,require()方法会将路径转换为真实的路径,并以真实路径作为索引,将编译执行后的结果放在缓存中,文件模块给Node 指明了确切的文件位置,所以查找过程中可以节约大量时间,其加载速度慢于核心模块

自定义模块

自定义模块指的是非核心模块,也不是路径形式的标识符。它是一种特殊的文件模块,可能是一个文件或者包的形式,这类模块是加载最慢的一种。

*** Node 定位文件模块的查找策略*** 具体表现为一个路径组成的数组。可以手动尝试一下:

在任意目录下执行输入node进入node环境,然后输入module.paths

在mac 下会得到下面一个数组输出


> module.paths

[ '/Users/fanrongrong/repl/node_modules',

  '/Users/fanrongrong/node_modules',

  '/Users/node_modules',

  '/node_modules',

  '/Users/fanrongrong/.node_modules',

  '/Users/fanrongrong/.node_libraries',

  '/Users/fanrongrong/.nvm/versions/node/v8.9.3/lib/node' ]

>

模块路径的生成规则

  • 当前文件目录下的node_modules目录

  • 父目录下的node_modules目录

  • 父目录的父目录下的node_modules目录,沿着路径逐级递归,直到跟目录

  • mac 会查找用户模块下的.node_modules和.node_libraries 目录,window会查找环境变量$HOME下的这两个目录

  • node 的安装目录下的node_modules (全局安装的包默认在这里,可以通过npm root -g查看路径)

文件定位

文件扩展名分析

require()在分析标识符的过程中,允许在标识符中不包含文件扩展名,这种情况下Node会按照.js、.json、.node的次序补足扩展名

目录分析和包

在分析标识符的过程中,require()通过分析文件扩展名之后,可能没有查找到对应的文件却得到了一个目录,此时Node会将目录当作一个包来处理。

包处理规则:Node在当前目录查找package.json文件,通过JSON.parse()解析出包的描述对象,从中取出main属性指定的文件名进行定位,如果文件名缺少扩展名,会进入扩展名分析步骤,如果main指定的文件名错误,或没有package.json文件,Node会将index当作默认文件名,然后依次查找index.js、index.json、index.node,如果目录分析中没有定位到任何文件,在自定义模块进入下一个模块的路径分析,如果模块的路径数组都遍历完依然没有找的目标文件,则会抛出查找失败的异常。

模块编译

去看源码
Node会新建一个模块对象,然后根据路径载入并编译,对于不同的文件扩展名,载入的方式也不同。

  • .js 文件。通过fs模块同步读取文件后编译执行
  • .node 文件。这是用c++编写的扩展文件,通过dlopen()方法加载最后编译生成的文件
  • .json文件。通过fs模块同步读取文件后,用JSON.parse()解析返回结果
  • 其余文件均当作js文件引入

每一个编译成功的模块都会将其文件路径作为索引缓存在Module._cache对象上。
其中,Module._extensions 会被赋值给require()extensions属性。通过console(require.extensions)可以查看已有的加载方式;

JavaScript模块的编译

在编译过程中,Node对获取的JavaScript 文件内容进行包装,如下:

(function (exports, require, module,  __filename, __dirname) {
  var math = require('math')
  exports.area = function () {
    return Math.PI * radius * radius2
  }
})

包与NPM

包结构

  • package.json: 包描述文件
  • bin : 存放可执行二进制文件目录
  • lib: 存放JavaScript 代码的目录

包描述文件

  • main 模块引入方法require方法会优先检查这个字段,并将它入口,如果不存在就查找index.js、index.node、index.json
  • bin 配置好bin 字段当npm install 包名 -g 时可以将脚本添加到执行路径中

安装依赖包

全局安装模式
  • 根据bin字段的配置,将实际的脚本链接到与node 可执行文件相同的路径下:
"bin": {
  "express": "./bin/express"
}

通过全局安装的模块都会被安装到一个统一目录下,一搬都是node的安装目录下的lib/node_modules 下

node的包引用

node commonjs 规范引用包方式

node 在8.0版之前都是遵循commenjs 规范进行的包引用(exports/module.exprots, require)
exports是module.exports 的一个引用,所以导出时可以使用exports.xxxx = xxx 的方式,而不能使用exports = xxxx的方式,可以使用module.exports = xxx的方式
这种方式是运行时加载,换句话说是在 NodeJS 脚本执行时才加载进来

node 8.0 之后加入ES方式的引用包方式(import, export)

使用ES方式需要在启动node时加入参数 --experimental-modules
这种方式引用是在静态分析时候就确定了引用关系,就像目标模块建立了一个符号链接,或者说建立了一个指针。这种加载方式加载效率应该略高于 CommonJS。
例子:

// a.js
var n = 3;
exports.n = 3
exports.add = () => {
  n++;
}
// b.js
var mod = require('./a.js');
console.log(mod.n);
mod.add();
console.log(mod.n);
// a.mjs
export let n = 3;
export let add = () => {
  n++;
}
// b.mjs
import {
    n,
    add
} from './a.mjs'
console.log(n);
add();
console.log(n);
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,390评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,821评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,632评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,170评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,033评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,098评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,511评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,204评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,479评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,572评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,341评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,213评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,576评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,893评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,171评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,486评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,676评论 2 335

推荐阅读更多精彩内容

  • nodejs 模块机制 简单模块定义和使用 在Node.js中,定义一个模块十分方便。我们以计算圆形的面积和周长两...
    艾伦先生阅读 1,039评论 0 4
  • 说明:该学习笔记参考《深入浅出Node.js》在学习过程中,添加了自己的理解和适当的补充!仅供参考! NodeJs...
    秋意思寒阅读 686评论 0 1
  • 之前 ECMAScript 的问题: 没有模块系统,标准库较少(如文件系统等缺失API),没有标准接口,无包管理系...
    Air_cc阅读 318评论 0 0
  • 早在Netscape诞生不久后,JavaScript就一直在探索本地编程的路,Rhino是其代表产物。无奈那时服务...
    itsmyturn阅读 311评论 0 0
  • 1.为什么要CommonJS规范 javascript存在的缺点 没有模块系统 标准库比较少 没有标准接口 缺乏包...
    maikuraki阅读 176评论 0 1