Node: 模块
我们知道,Node.js 选用 JavaScript 语言来编写代码。JavaScript 这门语言呢,之前主要用于前端应用,并没有相应的模块管理功能,而是以 script
标签为单位,直接引入即可运行。Node.js 主要运行在后端,这怎么办呢?好在它借鉴了 CommonJS
中的 Modules
规范,实现了一套易用的模块系统。今天,我们就来介绍一下 Node.js 中的模块。
举个栗子
下面我们写一段代码,封装一个模块,这个模块包含一个方法,对指定的人返回一句问候语:
// greeting.js
// 问候一声
exports.greet = (name) => {
return `hello ${name}`;
};
看上去挺简单的,直接在 exports
上面定义一个 greet
方法,即可将该方法暴露出去。
然后,我们在主文件中引入该模块,并调用它的 greet
方法:
// main.js
const greeting = require('./greeting');
console.log(greeting.greet('Scott'));
在引入该模块时,我们使用了相对路径 ./greeting
,Node.js 运行时会根据这个路径,找到我们自定义模块文件,然后编译执行。
最后,在命令行中运行 main.js
,程序会输出运行结果:
$ node main.js
hello Scott
一探究竟
下面,我们就来分析一下 Node.js 中的模块系统。
Node.js 中的模块分为 内置模块
和 自定义模块
两大类,而引用一个模块时一般会经历 路径分析
、文件定位
和 编译执行
三个步骤。
需要注意的是,如果我们引入内置模块,只需要 路径分析
这一个步骤,这是因为,内置模块在 Node.js 进程启动时,就已经被加载进内存了,可以直接引用,并且优先进行路径分析,所以不再需要 文件定位
和 编译执行
了。
引入内置模块时,只需指定模块名即可:
const http = require('http');
如果是引入自定义模块,则包含以下几种形式:
- 使用
.
或..
起始的相对路径 - 使用
/
起始的绝对路径 - 使用第三方模块名
如果使用了 相对路径
和 绝对路径
,require(module)
会根据模块路径读取相应的文件,然后编译执行,并将结果放入缓存,下次引入时直接从缓存中取结果。
如果我们引入一个第三方模块,例如 connect
模块,这时候,我们的引入方式和内置核心模块是类似的:
const connect = require('connect');
首先,Node.js 会先按内置模块查找,但发现它并不在内置模块名单中,所以接下来,要在 模块路径
中查找该模块对应的包。
这里提到了一个概念:模块路径
。它是 Node.js 定位文件模块的一种查找策略,表现形式是包含多个路径的一个数组。
为了验证这个数据结构,我们在代码中添加下面一行:
console.log(module.paths);
运行代码后,我们会得到下面输出内容:
[ '/Users/Scott/learning/node_modules',
'/Users/Scott/node_modules',
'/Users/node_modules',
'/node_modules' ]
可以看到,模块路径列举了这些目录:当前目录下的 node_modules 目录、父级目录下的 node_modules 目录、祖先目录下的 node_modules 目录、根目录下的 node_modules 目录。
在加载过程中,Node.js 会按照这个路径数组,逐个进行尝试,直到找到目标模块文件或包为止。
这里需要提醒一下,在使用 require(module)
时,如果参数不包含后缀名,Node.js 会按照 .js
、.node
、.json
次序补足后缀名,然后逐个尝试以单线程同步形式加载,所以,对于非 .js
后缀名的文件,引入时最好加上后缀名,以提高加载的速度。
回到我们最开始的一个小例子,这个程序中的 greeting.js
模块,通过 exports.greet
的形式导出一个方法,为什么能这样写呢,exports
是从哪里来的呢?
原来,Node.js 在加载一个模块时,会首先对它进行编译,在这个过程中,进行了一次头尾的包装,我们上面例子中的模块,经过包装之后是这个样子的:
(function (exports, require, module, __filename, __dirname) {
exports.greet = (name) => {
return `hello ${name}`;
};
});
前面我们直接拿来用的 exports
、require
以及 module
,原来是这么来的,还有 __filename
和 __dirname
,它们并不是全局变量,而是由包装函数传递进来的,表示当前模块文件的路径和目录。
经过上面的包装,我们的每个模块都有了自己的作用域,包装之后的代码会由 vm
原生模块的 runInThisContext()
方法执行,同时将包装函数所需的参数传递进去。
默认情况下,exports
和 module.exports
都指向一个空对象引用,即:
module.exports = exports = {};
所以上面我们给 exports
添加了一个方法,同时也会改变 module.exports
。
当然,我们也可以使用下面这种方式,改变导出的引用对象:
module.exports = {
greet: (name) => {
return `hello ${name}`;
}
};
但注意,不要在模块内使用下面这种方式:
// 无效的导出方式
exports = {
greet: (name) => {
return `hello ${name}`;
}
};
原因在于它更改的只是形参的引用,却对实参引用没有做任何更改,所以是无效的导出方式。
Node: 模块的更多相关文章
- node模块加载层级优化
模块加载痛点 大家也或多或少的了解node模块的加载机制,最为粗浅的表述就是依次从当前目录向上级查询node_modules目录,若发现依赖则加载.但是随着应用规模的加大,目录层级越来越深,若是在某个 ...
- node模块系统常用命令
node模块系统常用命令 命令 示例 备注 安装模块 npm install commander 最新版本 npm install commander@1.0.0 指定版本 npm install c ...
- Commonjs规范及Node模块实现
前面的话 Node在实现中并非完全按照CommonJS规范实现,而是对模块规范进行了一定的取舍,同时也增加了少许自身需要的特性.本文将详细介绍NodeJS的模块实现 引入 nodejs是区别于java ...
- 模块机制 之commonJs、node模块 、AMD、CMD
在其他高级语言中,都有模块中这个概念,比如java的类文件,PHP有include何require机制,JS一开始就没有模块这个概念,起初,js通过<script>标签引入代码的方式显得杂 ...
- NW.js安装原生node模块node-printer控制打印机
1.安装原生node模块 #全局安装nw-gyp npm install -g nw-gyp #设置目标NW.js版本 set npm_config_target=0.31.4 #设置构建架构,ia3 ...
- 深入了解Node模块原理
深入了解Node模块原理 当我们编写JavaScript代码时,我们可以申明全局变量: var s = 'global'; 在浏览器中,大量使用全局变量可不好.如果你在a.js中使用了全局变量s,那么 ...
- 【转】Commonjs规范及Node模块实现
前言: Node在实现中并非完全按照CommonJS规范实现,而是对模块规范进行了一定的取舍,同时也增加了少许自身需要的特性.本文将详细介绍NodeJS的模块实现 引入 nodejs是区别于javas ...
- [转]模块化——Common规范及Node模块实现
Node在实现中并非完全按照CommonJS规范实现,而是对模块规范进行了一定的取舍,同时也增加了少许自身需要的特性.本文将详细介绍NodeJS的模块实现 引入 nodejs是区别于javascrip ...
- node 模块正确暴露方法
一个node模块,为了能够服用,就需要将其暴露,那么如何正确写呢?(参考:https://developer.mozilla.org/zh-CN/docs/Learn/Server-side/Expr ...
- Electron结合React,在渲染进程中使用 node 模块
Electron结合React,在渲染进程中使用 node 模块 问题 将create-react-app与electron集成在了一个项目中.但是在React中无法使用electron.当在Reac ...
随机推荐
- HDFS java API TROUBLESHOOTING
官方文档:https://hadoop.apache.org/docs/r2.9.2/hadoop-project-dist/hadoop-common/SingleCluster.html 配置免密 ...
- 转 oracle 正则表达式和查询最大文件号 SQL
###sample 1 https://www.cnblogs.com/lxl57610/p/8227599.html Oracle使用正则表达式离不开这4个函数: 1.regexp_like 2.r ...
- SSAS 项目部署失败的问题
在创建SSAS项目过程中,创建数据源.数据源视图.多维数据集.纬度等一切都没有问题.但是在“进程”这一步的时候,发现总是报错,提示如下.OLE DB 错误: OLE DB 或 ODBC 错误 : 用户 ...
- 《设计模式》读懂UML类图
一.类中的主要关系 继承.实现.组合.聚合.依赖.关联 二.UML类图 三.代码实现 public class H2O { } public class O2 { } public interface ...
- C#中Control的Invoke和BeginInvoke是相对于支线线程
近日,被Control的Invoke和BeginInvoke搞的头大,就查了些相关的资料,整理如下. Control的Invoke和BeginInvoke 是相对于支线线程(因为一般在支线线程中调用, ...
- JVM知识点总览-高级Java工程师面试必备
jvm 总体梳理 jvm体系总体分四大块: 类的加载机制 jvm内存结构 GC算法 垃圾回收 GC分析 命令调优 当然这些知识点在之前的文章中都有详细的介绍,这里只做主干的梳理 这里画了一个思维导图, ...
- 复杂的sql参考(3)
SELECT apply.assets_code, apply.loan_apply_code, cust.cust_name, cust.id_no, cust.mobile, platform.p ...
- Word页眉实现首页不同、奇偶页不同 、更改页眉横线、页眉文字对齐 -- 视频教程(8)
1. 目标 目标1:实现页眉"首页不同,奇偶页不同" 目标2:更改页眉横线 目标3:页眉文字有三部分:第一部分左对齐,第二部分居中,第三部分右对齐 2. 教程 未完 ...... ...
- 洛谷P2048 [NOI2010]超级钢琴 题解
2019/11/14 更新日志: 近期发现这篇题解有点烂,更新一下,删繁就简,详细重点.代码多加了注释.就酱紫啦! 正解步骤 我们需要先算美妙度的前缀和,并初始化RMQ. 循环 \(i\) 从 \(1 ...
- BJFU-216-基于链式存储结构的图书信息表的修改
#include<stdio.h> #include<stdlib.h> #define MAX 100 typedef struct Book{ double no; cha ...