精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

Node中如何引入一個(gè)模塊及其細(xì)節(jié)

開(kāi)發(fā) 前端
雖然它們?cè)谄匠J褂弥袃H僅是引入與導(dǎo)出模塊,但稍稍深入,便可見(jiàn)乾坤之大。在業(yè)界可用它們做一些比較 trick 的事情,雖然我不大建議使用這些黑科技,但稍微了解還是很有必要。

 在 node 環(huán)境中,有兩個(gè)內(nèi)置的全局變量無(wú)需引入即可直接使用,并且無(wú)處不見(jiàn),它們構(gòu)成了 nodejs 的模塊體系: module 與 require。以下是一個(gè)簡(jiǎn)單的示例 

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

雖然它們?cè)谄匠J褂弥袃H僅是引入與導(dǎo)出模塊,但稍稍深入,便可見(jiàn)乾坤之大。在業(yè)界可用它們做一些比較 trick 的事情,雖然我不大建議使用這些黑科技,但稍微了解還是很有必要。

  1.  如何在不重啟應(yīng)用時(shí)熱加載模塊?如 require 一個(gè) json 文件時(shí)會(huì)產(chǎn)生緩存,但是重寫文件時(shí)如何 watch
  2.  如何通過(guò)不侵入代碼進(jìn)行打印日志
  3.  循環(huán)引用會(huì)產(chǎn)生什么問(wèn)題?

module wrapper

當(dāng)我們使用 node 中寫一個(gè)模塊時(shí),實(shí)際上該模塊被一個(gè)函數(shù)包裹,如下所示: 

  1. (function(exports, require, module, __filename, __dirname) {  
  2.   // 所有的模塊代碼都被包裹在這個(gè)函數(shù)中  
  3.   const fs = require('fs')  
  4.   const add = (x, y) => x + y 
  5.   module.exports = add  
  6. }); 

因此在一個(gè)模塊中自動(dòng)會(huì)注入以下變量:

  •  exports
  •  require
  •  module
  •  __filename
  •  __dirname

module

調(diào)試最好的辦法就是打印,我們想知道 module 是何方神圣,那就把它打印出來(lái)! 

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

  •  module.id: 如果是 . 代表是入口模塊,否則是模塊所在的文件名,可見(jiàn)如下的 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ù)源碼很容易理解一個(gè)模塊中的變量:

  •  exports: 實(shí)際上是 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

通過(guò) node 的 REPL 控制臺(tái),或者在 VSCode 中輸出 require 進(jìn)行調(diào)試,可以發(fā)現(xiàn) require 是一個(gè)極其復(fù)雜的對(duì)象

require

從以上 module wrapper 的源碼中也可以看出 require 由 makeRequireFunction 函數(shù)生成,如下 

  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 實(shí)際上是 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;  

    ❝    關(guān)于 require 更詳細(xì)的信息可以去參考官方文檔: Node API: require[2]    ❞

require(id)

require 函數(shù)被用作引入一個(gè)模塊,也是平常最常見(jiàn)最常用到的函數(shù) 

  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 引入一個(gè)模塊時(shí),實(shí)際上通過(guò) Module._load 載入,大致的總結(jié)如下:

  1.  如果 Module._cache 命中模塊緩存,則直接取出 module.exports,加載結(jié)束
  2.  如果是 NativeModule,則 loadNativeModule 加載模塊,如 fs、http、path 等模塊,加載結(jié)束
  3.  否則,使用 Module.load 加載模塊,當(dāng)然這個(gè)步驟也很長(zhǎng),下一章節(jié)再細(xì)講 
  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

「當(dāng)代碼執(zhí)行 require(lib) 時(shí),會(huì)執(zhí)行 lib 模塊中的內(nèi)容,并作為一份緩存,下次引用時(shí)不再執(zhí)行模塊中內(nèi)容」。

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

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

這里有個(gè)小測(cè)試:

    ❝    有兩個(gè)文件: index.js 與 utils.js。utils.js 中有一個(gè)打印操作,當(dāng) index.js 引用 utils.js 多次時(shí),utils.js 中的打印操作會(huì)執(zhí)行幾次。代碼示例如下    ❞

「index.js」 

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

「utils.js」 

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

「答案是只執(zhí)行了一次」,因此 require.cache,在 index.js 末尾打印 require,此時(shí)會(huì)發(fā)現(xiàn)一個(gè)模塊緩存 

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

那回到本章剛開(kāi)始的問(wèn)題:

    ❝    如何不重啟應(yīng)用熱加載模塊呢?    ❞

答:「刪掉 Module._cache」,但同時(shí)會(huì)引發(fā)問(wèn)題,如這種 一行 delete require.cache 引發(fā)的內(nèi)存泄漏血案[3]

所以說(shuō)嘛,這種黑魔法大幅修改核心代碼的東西開(kāi)發(fā)環(huán)境玩一玩就可以了,千萬(wàn)不要跑到生產(chǎn)環(huán)境中去,畢竟黑魔法是不可控的。

總結(jié)

  1. 模塊中執(zhí)行時(shí)會(huì)被 module wrapper 包裹,并注入全局變量 require 及 module 等
  2.  module.exports 與 exports 的關(guān)系實(shí)際上是 exports = module.exports
  3.  require 實(shí)際上是 module.require
  4.  require.cache 會(huì)保證模塊不會(huì)被執(zhí)行多次
  5.  不要使用 delete require.cache 這種黑魔法 

 

責(zé)任編輯:龐桂玉 來(lái)源: 前端大全
相關(guān)推薦

2020-08-24 08:07:32

Node.js文件函數(shù)

2021-06-09 07:55:19

NodeEventEmitte驅(qū)動(dòng)

2017-06-20 12:48:55

React Nativ自定義模塊Note.js

2015-10-12 16:45:26

NodeWeb應(yīng)用框架

2024-03-26 10:38:47

模塊CommonJSES

2021-07-06 14:36:05

RustLinux內(nèi)核模塊

2021-03-08 10:49:11

漏洞攻擊網(wǎng)絡(luò)安全

2021-03-13 12:54:50

Node進(jìn)程Cron

2011-07-18 13:34:44

SQL Server數(shù)拼接字符串

2019-09-10 09:12:54

2012-08-07 11:28:13

卸載linux

2024-04-11 08:30:05

JavaScript數(shù)組函數(shù)

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前端

2021-03-11 12:19:39

Linux運(yùn)維Linux系統(tǒng)

2011-08-29 15:12:24

UbuntuLinux模塊
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

粉嫩91精品久久久久久久99蜜桃| 女性裸体视频网站| 动漫一区二区在线| 51自拍视频在线观看| 天堂资源最新在线| 欧美成人嫩草网站| 欧美亚洲动漫另类| 欧美精品一区三区在线观看| 波多野结衣亚洲色图| 超碰这里只有精品| 99久久精品国产一区二区三区| 最近免费中文字幕视频2019| 国产二区视频在线播放| 国产色视频在线| 日韩免费一区| 在线观看欧美精品| 免费看污久久久| 精品视频在线观看免费| 国产视频网站一区二区三区| 国产精品少妇自拍| 国产99视频精品免视看7| 999精品免费视频| jizz一区二区三区| 国产成a人亚洲精品| 美女精品视频一区| 一级 黄 色 片一| 国产在线激情| 国产一区欧美日韩| www.日韩不卡电影av| 热久久精品免费视频| 国际av在线| 久久中文在线| 国产亚洲欧美另类中文| 凹凸日日摸日日碰夜夜爽1| 婷婷在线观看视频| 亚洲美洲欧洲综合国产一区| 亚洲国产精品久久| 缅甸午夜性猛交xxxx| 日本激情一区二区三区| 亚洲性图久久| 国产丝袜精品视频| mm1313亚洲国产精品无码试看| 欧美资源一区| 69久久精品无码一区二区| 免费网站看v片在线a| 国模少妇一区二区三区| 久久久精品国产网站| 亚洲天堂网站在线| 国产桃色电影在线播放| 97久久精品人人爽人人爽蜜臀| 日本精品视频在线| 精品熟妇无码av免费久久| 亚洲一区导航| 亚洲国产sm捆绑调教视频| 精品在线视频一区二区| 久久精品久久久久久久| 日本精品黄色| 日韩视频中午一区| 成人午夜精品久久久久久久蜜臀| 五月婷婷深深爱| 免费国产亚洲视频| 欧美精品日韩www.p站| 性久久久久久久久久久| 日本在线中文字幕一区二区三区| **性色生活片久久毛片| 国产精品一区二区免费| 无码视频一区二区三区| 亚洲五月综合| 日韩国产精品一区| 久久久久久蜜桃一区二区| 中文字幕有码在线观看| 久久婷婷一区二区三区| 国产精品国产三级国产aⅴ浪潮 | 亚洲男人7777| 在线观看国产福利| av成人 com a| 国产欧美日韩不卡| 国产精品乱子乱xxxx| 国产精品传媒在线观看| 在线欧美三区| 色偷偷9999www| 国产精品无码一区二区三| 国产欧美视频一区| 中文字幕永久免费视频| 亚洲一级网站| 丝袜一区二区三区| 日韩精品一区二区三区高清免费| 日本一区二区三区中文字幕| 午夜精品久久一牛影视| 曰韩不卡视频| 激情视频在线观看免费| 成人免费不卡视频| 成人黄色激情网| av一级在线观看| 亚洲激情精品| 欧美成人国产va精品日本一级| 少妇人妻好深好紧精品无码| 国产色噜噜噜91在线精品| 欧美裸体bbwbbwbbw| 妞干网在线免费视频| 福利小视频在线| 亚洲免费在线播放| 亚洲最新免费视频| 成人免费视频| 久久夜色精品一区| 精品久久蜜桃| 免费av网站观看| 国产一区二区h| 国产欧美亚洲视频| 在线观看亚洲国产| 秋霞午夜鲁丝一区二区老狼| 日本精品视频在线观看| 日本一级片免费看| 91久久黄色| 久久理论片午夜琪琪电影网| 国产主播在线观看| 亚洲精品1区| 97超级碰在线看视频免费在线看| 国产奶水涨喷在线播放| 亚洲黄页一区| 欧美亚洲在线播放| 日本三级一区二区| 日韩亚洲国产精品| 国产91成人video| 日韩欧美一级视频| 三级欧美在线一区| 国产精品极品美女在线观看免费 | 人妻换人妻a片爽麻豆| 日韩一区二区三区高清在线观看| 欧美一级片免费看| 无码人妻久久一区二区三区蜜桃| 久久久久毛片免费观看| 精品国产一区二区三区久久影院| 精品无码av一区二区三区不卡| 国产精品亚洲综合在线观看 | 图片婷婷一区| 亚洲精品小视频| 国产美女永久免费无遮挡| av中文一区| 久久综合电影一区| 精品少妇爆乳无码av无码专区| 伊人成人在线| 91av在线播放| 夜夜嗨aⅴ一区二区三区| 韩国毛片一区二区三区| 国产chinese精品一区二区| 欧美一级特黄aaaaaa| 99久久99久久久精品齐齐| 免费看成人午夜电影| 伊人免费在线| 亚洲午夜av在线| aⅴ在线免费观看| 精品美女一区| 精品欧美久久久| 国产伦精品一区二区三区妓女 | 成人av在线网站| 欧美极品色图| 国产美女福利在线| 黄色一区二区在线观看| 爆乳熟妇一区二区三区霸乳| 成人在线日韩| 日韩精品免费视频| 成人一级黄色大片| 99国产精品自拍| 成人国产精品一区二区| 香蕉久久一区二区三区| 国产精品传媒视频| 337p粉嫩大胆噜噜噜鲁| 久久亚洲资源中文字| 亚洲成人亚洲激情| 一区二区三区在线播放视频| 亚洲高清不卡| 成人国产精品久久久| 飘雪影院手机免费高清版在线观看 | 国产精品www| 性生交生活影碟片| 欧美国产精品一区二区三区| 野外做受又硬又粗又大视频√| 国产精品久久亚洲不卡| 亚洲成年人在线| 午夜激情福利网| 日韩av电影天堂| 狠狠色综合色区| a级在线观看| 欧美日韩一区二区三区免费看| 给我免费观看片在线电影的| 综合久久一区| 国产精品视频99| 欧美性孕妇孕交| 亚洲国产欧美日韩另类综合| 亚洲最大天堂网| 精品美女久久| 欧美最猛性xxxx| 色呦呦免费观看| 一区二区不卡在线播放| 国模私拍视频在线观看| 精品成人影院| 欧美资源在线观看| 四虎国产精品永远| 亚洲国产成人高清精品| 在线观看视频在线观看| 91蜜臀精品国产自偷在线 | 成人午夜毛片| 国产香蕉精品视频一区二区三区| 久久久美女视频| 国产精品系列在线观看| 亚洲免费av网| 91丨精品丨国产| 波霸ol色综合久久| 国产精品国产三级国产aⅴ | 欧美人妻一区二区| 激情图区综合网| 一区二区国产日产| 青青草国产一区二区三区| 在线观看亚洲视频| 一二三四区视频| 中文字幕一区二区三区不卡| 污色网站在线观看| 日韩电影免费网址| 国产精品综合网站| 毛片av在线| 337p亚洲精品色噜噜噜| 懂色av懂色av粉嫩av| 国产精品综合在线视频| 国产欧美久久久久| 精品视频在线你懂得| 91a在线视频| 国产对白叫床清晰在线播放| 欧美日韩高清影院| 欧美日韩三级在线观看| 国产精品影视天天线| 国产xxxx振车| 丝袜连裤袜欧美激情日韩| 欧美在线视频在线播放完整版免费观看| 欧美日韩视频精品二区| 欧美日韩一区二区三区不卡| 一级片一级片一级片| 国产电影精品久久禁18| 狠狠97人人婷婷五月| 欧美偷拍自拍| 亚洲自拍偷拍区| √8天堂资源地址中文在线| 日韩精品在线观看视频| 中文字幕乱伦视频| 亚洲乱码一区二区三区在线观看| 亚洲色图欧美另类| 久久久久久穴| 一区二区不卡在线观看| japanese色系久久精品| 68精品国产免费久久久久久婷婷| 国产中文在线视频| 欧美久久一区二区| 在线观看 中文字幕| 中文字幕免费一区| 国产精品果冻传媒| 日韩高清在线观看| 男女啪啪免费观看| 亚洲制服一区| 91天堂在线观看| yellow在线观看网址| 少妇激情综合网| 男人天堂网在线视频| 欧美色老头old∨ideo| 久久国产露脸精品国产| 久久久久高清精品| 被黑人猛躁10次高潮视频| 另类天堂av| 欧美黄网在线观看| 精品福利久久久| 国产伦精品一区二区三区照片| 69堂免费精品视频在线播放| 久久青草福利网站| 午夜毛片在线| 国产网站欧美日韩免费精品在线观看| 一本色道久久综合精品婷婷 | 69国产精品视频免费观看| 亚洲桃色在线一区| 这里只有久久精品| 国产不卡视频在线播放| 亚欧在线免费观看| 亚洲福利一区| 精品国产一区二区三区在线| 欧美日韩中文一区二区| 激情久久av| silk一区二区三区精品视频| 91精品久久久久久久| 黑人巨大精品| 5252色成人免费视频| 国产盗摄一区二区| 欧美久久久精品| 亚洲欧美视频一区二区| 亚洲人成在线免费观看| 欧美一区二区三区成人片在线| 日韩一区二区三区电影在线观看| 伊人网综合在线| 91国偷自产一区二区三区观看| 日韩人妻无码一区二区三区99| 一区二区三区小说| 成人免费精品动漫网站| 国产精品三级电影| av手机在线播放| 久久精品一级爱片| 无码国产69精品久久久久同性| 91在线播放网址| 人妻无码中文久久久久专区| 不卡的av电影| 欧洲熟妇的性久久久久久| 国产精品一卡二卡在线观看| 超碰中文字幕在线观看| 国产在线国偷精品免费看| 亚洲美女性囗交| 狠狠色综合日日| 国产又粗又长又爽又黄的视频| 久久99精品国产麻豆婷婷| 国产成年人视频网站| 国产一区91精品张津瑜| 亚洲国产欧美91| 成人综合婷婷国产精品久久| 亚洲一区二区三区四区av| 99久久综合色| 91精品人妻一区二区三区蜜桃欧美| 久久综合给合久久狠狠狠97色69| 亚洲黄色在线网站| 国产无人区一区二区三区| 久久亚洲AV无码专区成人国产| 中文字幕不卡一区| 亚洲一区电影在线观看| 亚洲人成影院在线观看| 成人性生活毛片| 一区二区三区四区国产精品| 久一区二区三区| 天天色天天操综合| 黄色片中文字幕| 欧美视频一区在线| 国产乱码一区二区| 欧美精品一区二区精品网| 青青草超碰在线| 最近免费中文字幕视频2019| caopon在线免费视频| 国语自产精品视频在线看| 成人软件在线观看| 成人av在线天堂| 亚洲欧洲国产精品一区| 裸模一区二区三区免费| 97精品97| 欧美日韩精品在线一区二区 | www.av网站| 日韩国产中文字幕| 97视频精彩视频在线观看| 久久久91精品国产一区不卡| 春色校园综合激情亚洲| 国产精品久久久久久久久久| 国产区一区二| 欧美国产二区| 综合久久久久| 亚洲 中文字幕 日韩 无码| 国产真实精品久久二三区| 午夜免费福利影院| 国产精品无圣光一区二区| 久久久久久久久久久久久久久久久| 欧美日韩在线视频一区二区| 一级爱爱免费视频| 亚洲第一区中文99精品| 亚洲欧美视频一区二区| 97视频免费在线看| 四虎成人精品一区二区免费网站| 国产精品我不卡| 日韩欧美国产精品综合嫩v| 国产精品又粗又长| 久久精品72免费观看| 99re久久精品国产| 国产精品电影院| 在线能看的av| 日韩三级中文字幕| 国产午夜视频在线观看| 美女视频黄免费的亚洲男人天堂| 性欧美18xxxhd| 亚洲a在线播放| 成人激情免费视频| 国产二区视频在线播放| 国产成人亚洲综合色影视| 日韩精品一区二区三区中文不卡| 中文字幕人妻精品一区| 精品1区2区在线观看| 欧美成人hd| 国产精品91在线| 女同久久另类99精品国产| 男人天堂成人网| 日本不卡高清视频| 日韩网站在线播放| 亚洲午夜精品久久久久久久久| 国产精品免费无遮挡| 国产一区二区三区精品久久久| bl在线肉h视频大尺度| 国产欧美最新羞羞视频在线观看| 天堂俺去俺来也www久久婷婷| 久久香蕉视频网站| 久久精品99国产精品| 日本xxxxxxxxx18| 欧美性猛交xxxx乱大交蜜桃|