Node中如何引入一个模块及其细节

开发 前端
虽然它们在平常使用中仅仅是引入与导出模块,但稍稍深入,便可见乾坤之大。在业界可用它们做一些比较 trick 的事情,虽然我不大建议使用这些黑科技,但稍微了解还是很有必要。

 在 node 环境中,有两个内置的全局变量无需引入即可直接使用,并且无处不见,它们构成了 nodejs 的模块体系: module 与 require。以下是一个简单的示例 

  1. const fs = require('fs')  
  2. const add = (x, y) => x + y  
  3. module.exports = add 

虽然它们在平常使用中仅仅是引入与导出模块,但稍稍深入,便可见乾坤之大。在业界可用它们做一些比较 trick 的事情,虽然我不大建议使用这些黑科技,但稍微了解还是很有必要。

  1.  如何在不重启应用时热加载模块?如 require 一个 json 文件时会产生缓存,但是重写文件时如何 watch
  2.  如何通过不侵入代码进行打印日志
  3.  循环引用会产生什么问题?

module wrapper

当我们使用 node 中写一个模块时,实际上该模块被一个函数包裹,如下所示: 

  1. (function(exports, require, module, __filename, __dirname) {  
  2.   // 所有的模块代码都被包裹在这个函数中  
  3.   const fs = require('fs')  
  4.   const add = (x, y) => x + y 
  5.   module.exports = add  
  6. }); 

因此在一个模块中自动会注入以下变量:

  •  exports
  •  require
  •  module
  •  __filename
  •  __dirname

module

调试最好的办法就是打印,我们想知道 module 是何方神圣,那就把它打印出来! 

  1. const fs = require('fs')  
  2. const add = (x, y) => x + y  
  3. module.exports = add  
  4. console.log(module) 

  •  module.id: 如果是 . 代表是入口模块,否则是模块所在的文件名,可见如下的 koa
  •  module.exports: 模块的导出

koa module

module.exports 与 exports

    ❝    `module.exports` 与 `exports` 有什么关系?[1]    ❞

从以下源码中可以看到 module wrapper 的调用方 module._compile 是如何注入内置变量的,因此根据源码很容易理解一个模块中的变量:

  •  exports: 实际上是 module.exports 的引用
  •  require: 大多情况下是 Module.prototype.require
  •  module
  •  __filename
  •  __dirname: path.dirname(__filename) 
  1. // <node_internals>/internal/modules/cjs/loader.js:1138  
  2. Module.prototype._compile = function(content, filename) {  
  3.   // ...  
  4.   const dirname = path.dirname(filename);  
  5.   const require = makeRequireFunction(this, redirects);  
  6.   let result;  
  7.   // 从中可以看出:exports = module.exports  
  8.   const exports = this.exports;  
  9.   const thisValue = exports 
  10.   const module = this 
  11.   if (requireDepth === 0) statCache = new Map();  
  12.   if (inspectorWrapper) {  
  13.     result = inspectorWrapper(compiledWrapper, thisValue, exports,  
  14.                               require, module, filename, dirname);  
  15.   } else {  
  16.     result = compiledWrapper.call(thisValue, exports, require, module,  
  17.                                   filename, dirname);  
  18.   }  
  19.   // ...  

require

通过 node 的 REPL 控制台,或者在 VSCode 中输出 require 进行调试,可以发现 require 是一个极其复杂的对象

require

从以上 module wrapper 的源码中也可以看出 require 由 makeRequireFunction 函数生成,如下 

  1. // <node_internals>/internal/modules/cjs/helpers.js:33  
  2. function makeRequireFunction(mod, redirects) {  
  3.   const Module = mod.constructor;  
  4.   let require;  
  5.   if (redirects) {  
  6.     // ...  
  7.   } else { 
  8.      // require 实际上是 Module.prototype.require  
  9.     require = function require(path) {  
  10.       return mod.require(path);  
  11.     };  
  12.   }  
  13.   function resolve(request, options) { // ... }  
  14.   require.resolve = resolve;  
  15.   function paths(request) {  
  16.     validateString(request, 'request');  
  17.     return Module._resolveLookupPaths(request, mod);  
  18.   }  
  19.   resolve.paths = paths;  
  20.   require.main = process.mainModule;  
  21.   // Enable support to add extra extension types.  
  22.   require.extensions = Module._extensions;  
  23.   require.cache = Module._cache;  
  24.   return require;  

    ❝    关于 require 更详细的信息可以去参考官方文档: Node API: require[2]    ❞

require(id)

require 函数被用作引入一个模块,也是平常最常见最常用到的函数 

  1. // <node_internals>/internal/modules/cjs/loader.js:1019  
  2. Module.prototype.require = function(id) { 
  3.    validateString(id, 'id');  
  4.   if (id === '') {  
  5.     throw new ERR_INVALID_ARG_VALUE('id', id,  
  6.                                     'must be a non-empty string');  
  7.   }  
  8.   requireDepth++;  
  9.   try {  
  10.     return Module._load(id, this, /* isMain */ false);  
  11.   } finally { 
  12.      requireDepth--;  
  13.   }  

而 require 引入一个模块时,实际上通过 Module._load 载入,大致的总结如下:

  1.  如果 Module._cache 命中模块缓存,则直接取出 module.exports,加载结束
  2.  如果是 NativeModule,则 loadNativeModule 加载模块,如 fs、http、path 等模块,加载结束
  3.  否则,使用 Module.load 加载模块,当然这个步骤也很长,下一章节再细讲 
  1. // <node_internals>/internal/modules/cjs/loader.js:879  
  2. Module._load = function(request, parent, isMain) {  
  3.   let relResolveCacheIdentifier;  
  4.   if (parent) {  
  5.     // ...  
  6.   }  
  7.   const filename = Module._resolveFilename(request, parent, isMain);  
  8.   const cachedModule = Module._cache[filename];  
  9.   // 如果命中缓存,直接取缓存  
  10.   if (cachedModule !== undefined) {  
  11.     updateChildren(parent, cachedModule, true);  
  12.     return cachedModule.exports;  
  13.   }  
  14.   // 如果是 NativeModule,加载它  
  15.   const mod = loadNativeModule(filename, request);  
  16.   if (mod && mod.canBeRequiredByUsers) return mod.exports;  
  17.   // Don't call updateChildren(), Module constructor already does.  
  18.   const module = new Module(filename, parent);  
  19.   if (isMain) {  
  20.     process.mainModule = module 
  21.     module.id = '.' 
  22.   }  
  23.   Module._cache[filename] = module;  
  24.   if (parent !== undefined) { // ... }  
  25.   let threw = true 
  26.   try {  
  27.     if (enableSourceMaps) {  
  28.       try {  
  29.         // 如果不是 NativeModule,加载它  
  30.         module.load(filename);  
  31.       } catch (err) {  
  32.         rekeySourceMap(Module._cache[filename], err);  
  33.         throw err; /* node-do-not-add-exception-line */  
  34.       }  
  35.     } else {  
  36.       module.load(filename);  
  37.     }  
  38.     threw = false 
  39.   } finally {  
  40.     // ...  
  41.   }  
  42.   return module.exports;  
  43. }; 

require.cache

「当代码执行 require(lib) 时,会执行 lib 模块中的内容,并作为一份缓存,下次引用时不再执行模块中内容」。

这里的缓存指的就是 require.cache,也就是上一段指的 Module._cache 

  1. // <node_internals>/internal/modules/cjs/loader.js:899  
  2. require.cache = Module._cache; 

这里有个小测试:

    ❝    有两个文件: index.js 与 utils.js。utils.js 中有一个打印操作,当 index.js 引用 utils.js 多次时,utils.js 中的打印操作会执行几次。代码示例如下    ❞

「index.js」 

  1. // index.js  
  2. // 此处引用两次  
  3. require('./utils')  
  4. require('./utils') 

「utils.js」 

  1. // utils.js  
  2. console.log('被执行了一次') 

「答案是只执行了一次」,因此 require.cache,在 index.js 末尾打印 require,此时会发现一个模块缓存 

  1. // index.js  
  2. require('./utils')  
  3. require('./utils')  
  4. console.log(require) 

那回到本章刚开始的问题:

    ❝    如何不重启应用热加载模块呢?    ❞

答:「删掉 Module._cache」,但同时会引发问题,如这种 一行 delete require.cache 引发的内存泄漏血案[3]

所以说嘛,这种黑魔法大幅修改核心代码的东西开发环境玩一玩就可以了,千万不要跑到生产环境中去,毕竟黑魔法是不可控的。

总结

  1. 模块中执行时会被 module wrapper 包裹,并注入全局变量 require 及 module 等
  2.  module.exports 与 exports 的关系实际上是 exports = module.exports
  3.  require 实际上是 module.require
  4.  require.cache 会保证模块不会被执行多次
  5.  不要使用 delete require.cache 这种黑魔法 

 

责任编辑:庞桂玉 来源: 前端大全
相关推荐

2020-08-24 08:07:32

Node.js文件函数

2021-06-09 07:55:19

NodeEventEmitte驱动

2017-06-20 12:48:55

React Nativ自定义模块Note.js

2015-10-12 16:45:26

NodeWeb应用框架

2024-03-26 10:38:47

模块CommonJSES

2021-07-06 14:36:05

RustLinux内核模块

2021-03-08 10:49:11

漏洞攻击网络安全

2021-03-13 12:54:50

Node进程Cron

2011-07-18 13:34:44

SQL Server数拼接字符串

2019-09-10 09:12:54

2012-08-07 11:28:13

卸载linux

2024-04-11 08:30:05

JavaScript数组函数

2016-12-07 17:45:44

Linux文件

2021-01-04 09:12:31

集合变量

2023-09-26 16:44:14

光模块

2014-08-01 10:24:11

2011-10-25 09:28:30

Node.js

2020-08-07 10:40:56

Node.jsexpress前端

2020-04-08 08:35:20

JavaScript模块函数

2011-08-29 15:12:24

UbuntuLinux模块
点赞
收藏

51CTO技术栈公众号