這篇文章將為大家詳細(xì)講解有關(guān)Node中怎么引入一個模塊,文章內(nèi)容質(zhì)量較高,因此小編分享給大家做個參考,希望大家閱讀完這篇文章后對相關(guān)知識有一定的了解。
從網(wǎng)站建設(shè)到定制行業(yè)解決方案,為提供成都網(wǎng)站設(shè)計、做網(wǎng)站服務(wù)體系,各種行業(yè)企業(yè)客戶提供網(wǎng)站建設(shè)解決方案,助力業(yè)務(wù)快速發(fā)展。創(chuàng)新互聯(lián)將不斷加快創(chuàng)新步伐,提供優(yōu)質(zhì)的建站服務(wù)。
module wrapper
當(dāng)我們使用 node 中寫一個模塊時,實際上該模塊被一個函數(shù)包裹,如下所示:
(function(exports, require, module, __filename, __dirname) { // 所有的模塊代碼都被包裹在這個函數(shù)中 const fs = require('fs') const add = (x, y) => x + y module.exports = add });
因此在一個模塊中自動會注入以下變量:
exports
require
module
__filename
__dirname
module
調(diào)試最好的辦法就是打印,我們想知道 module 是何方神圣,那就把它打印出來!
const fs = require('fs') const add = (x, y) => x + y module.exports = add console.log(module)
module.id: 如果是 . 代表是入口模塊,否則是模塊所在的文件名,可見如下的 koa
module.exports: 模塊的導(dǎo)出
koa module
module.exports 與 exports
? `module.exports` 與 `exports` 有什么關(guān)系?[1] ?
從以下源碼中可以看到 module wrapper 的調(diào)用方 module._compile 是如何注入內(nèi)置變量的,因此根據(jù)源碼很容易理解一個模塊中的變量:
exports: 實際上是 module.exports 的引用
require: 大多情況下是 Module.prototype.require
module
__filename
__dirname: path.dirname(__filename)
///internal/modules/cjs/loader.js:1138 Module.prototype._compile = function(content, filename) { // ... const dirname = path.dirname(filename); const require = makeRequireFunction(this, redirects); let result; // 從中可以看出:exports = module.exports const exports = this.exports; const thisValue = exports; const module = this; if (requireDepth === 0) statCache = new Map(); if (inspectorWrapper) { result = inspectorWrapper(compiledWrapper, thisValue, exports, require, module, filename, dirname); } else { result = compiledWrapper.call(thisValue, exports, require, module, filename, dirname); } // ... }
require
通過 node 的 REPL 控制臺,或者在 VSCode 中輸出 require 進行調(diào)試,可以發(fā)現(xiàn) require 是一個極其復(fù)雜的對象
require
從以上 module wrapper 的源碼中也可以看出 require 由 makeRequireFunction 函數(shù)生成,如下
///internal/modules/cjs/helpers.js:33 function makeRequireFunction(mod, redirects) { const Module = mod.constructor; let require; if (redirects) { // ... } else { // require 實際上是 Module.prototype.require require = function require(path) { return mod.require(path); }; } function resolve(request, options) { // ... } require.resolve = resolve; function paths(request) { validateString(request, 'request'); return Module._resolveLookupPaths(request, mod); } resolve.paths = paths; require.main = process.mainModule; // Enable support to add extra extension types. require.extensions = Module._extensions; require.cache = Module._cache; return require; }
? 關(guān)于 require 更詳細(xì)的信息可以去參考官方文檔: Node API: require[2] ?
require(id)
require 函數(shù)被用作引入一個模塊,也是平常最常見最常用到的函數(shù)
///internal/modules/cjs/loader.js:1019 Module.prototype.require = function(id) { validateString(id, 'id'); if (id === '') { throw new ERR_INVALID_ARG_VALUE('id', id, 'must be a non-empty string'); } requireDepth++; try { return Module._load(id, this, /* isMain */ false); } finally { requireDepth--; } }
而 require 引入一個模塊時,實際上通過 Module._load 載入,大致的總結(jié)如下:
鴻蒙官方戰(zhàn)略合作共建——HarmonyOS技術(shù)社區(qū)
如果 Module._cache 命中模塊緩存,則直接取出 module.exports,加載結(jié)束
如果是 NativeModule,則 loadNativeModule 加載模塊,如 fs、http、path 等模塊,加載結(jié)束
否則,使用 Module.load 加載模塊,當(dāng)然這個步驟也很長,下一章節(jié)再細(xì)講
///internal/modules/cjs/loader.js:879 Module._load = function(request, parent, isMain) { let relResolveCacheIdentifier; if (parent) { // ... } const filename = Module._resolveFilename(request, parent, isMain); const cachedModule = Module._cache[filename]; // 如果命中緩存,直接取緩存 if (cachedModule !== undefined) { updateChildren(parent, cachedModule, true); return cachedModule.exports; } // 如果是 NativeModule,加載它 const mod = loadNativeModule(filename, request); if (mod && mod.canBeRequiredByUsers) return mod.exports; // Don't call updateChildren(), Module constructor already does. const module = new Module(filename, parent); if (isMain) { process.mainModule = module; module.id = '.'; } Module._cache[filename] = module; if (parent !== undefined) { // ... } let threw = true; try { if (enableSourceMaps) { try { // 如果不是 NativeModule,加載它 module.load(filename); } catch (err) { rekeySourceMap(Module._cache[filename], err); throw err; /* node-do-not-add-exception-line */ } } else { module.load(filename); } threw = false; } finally { // ... } return module.exports; };
require.cache
「當(dāng)代碼執(zhí)行 require(lib) 時,會執(zhí)行 lib 模塊中的內(nèi)容,并作為一份緩存,下次引用時不再執(zhí)行模塊中內(nèi)容」。
這里的緩存指的就是 require.cache,也就是上一段指的 Module._cache
///internal/modules/cjs/loader.js:899 require.cache = Module._cache;
這里有個小測試:
? 有兩個文件: index.js 與 utils.js。utils.js 中有一個打印操作,當(dāng) index.js 引用 utils.js 多次時,utils.js 中的打印操作會執(zhí)行幾次。代碼示例如下 ?
「index.js」
// index.js // 此處引用兩次 require('./utils') require('./utils')
「utils.js」
// utils.js console.log('被執(zhí)行了一次')
「答案是只執(zhí)行了一次」,因此 require.cache,在 index.js 末尾打印 require,此時會發(fā)現(xiàn)一個模塊緩存
// index.js require('./utils') require('./utils') console.log(require)
那回到本章剛開始的問題:
? 如何不重啟應(yīng)用熱加載模塊呢? ?
答:「刪掉 Module._cache」,但同時會引發(fā)問題,如這種 一行 delete require.cache 引發(fā)的內(nèi)存泄漏血案[3]
所以說嘛,這種黑魔法大幅修改核心代碼的東西開發(fā)環(huán)境玩一玩就可以了,千萬不要跑到生產(chǎn)環(huán)境中去,畢竟黑魔法是不可控的。
關(guān)于Node中怎么引入一個模塊就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,可以學(xué)到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。