簡單模擬node.js中require的加載機制
一、先了解一下,nodejs中require的加載機制
1、require的加載文件順序
require 加載文件時可以省略擴展名:
require('./module');
// 此時文件按 JS 文件執(zhí)行
require('./module.js');
// 此時文件按 JSON 文件解析
require('./module.json');
// 此時文件預(yù)編譯好的 C++ 模塊執(zhí)行
require('./module.node');
// 載入目錄module目錄中的 package.json 中main指向的文件
require('./module/default.js');
// 載入目錄module 中的index.js文件
通過 ./ 或 ../ 開頭:則按照相對路徑從當(dāng)前文件所在文件夾開始尋找模塊;
require('../file.js');
=> 上級目錄下找 file.js 文件
通過 / 開頭:則以系統(tǒng)根目錄開始尋找模塊;
require('/Users/iceStone/Documents/file.js');
=> 以絕對路徑的方式找,沒有任何異議
如果參數(shù)字符串不以“./“ 或 ”/“ 開頭,則表示加載的是一個默認提供的核心模塊(位于 Node 的系統(tǒng)安裝目錄中):
require('fs');
=> 加載核心模塊中的文件系統(tǒng)模塊
或者從當(dāng)前目錄向上搜索 node_modules 目錄中的文件:
require('my_module');
=> 各級 node_modules 文件夾中搜索 my_module.js 文件;
如果 require 傳入的是一個目錄的路徑,會自動查看該目錄的 package.json 文件,然后加載 main 字段指定的入口文件
如果package.json文件沒有main字段,或者根本就沒有package.json文件,則默認找目錄下的 index.js 文件作為模塊:
require('./calcuator');
=> 當(dāng)前目錄下找 calculator 目錄中的 index.js 文件
2、require緩存
第一次加載某個模塊時,Node 會緩存該模塊。以后再加載該模塊,就直接從緩存取出該模塊的 module.exports 屬性(不會再次執(zhí)行該模塊)
如果需要多次執(zhí)行模塊中的代碼,一般可以讓模塊暴露行為(函數(shù)),模塊的緩存可以通過 require.cache 拿到,同樣也可以刪除
3、所有代碼都運行在模塊作用域,不會污染全局作用域。
二、模擬require函數(shù)
require的加載內(nèi)部比較復(fù)雜,下面讓我們進行簡單的模擬加載
require的簡單實現(xiàn)機制為:
將傳入的模塊id通過加載規(guī)則找到對應(yīng)的模塊文件
讀取這個文件里面的代碼
通過拼接方式為該段代碼構(gòu)建私有空間
執(zhí)行該代碼
拿到module.exports 返回
nodejs_require.js // [require函數(shù)模擬] "use strict" function $require(id) { //1、先查看模塊是否存在,如果不存在則拋出 Can't found file //2、如果存在,就讀取代碼 const fs = require('fs'); const path = require('path'); // 文件名 let filename = id; //查看是否是絕對路徑 if (!path.isAbsolute(filename)) { filename = path.join(__dirname, id); } // 文件目錄 let dirname = path.dirname(filename); //模擬require的緩存機制 $require.cache = $require.cache || {}; if($require.cache[filename]){ return $require.cache[filename].exports; } // 讀取模塊代碼 let code=""; try { code = fs.readFileSync(filename,'utf-8'); // 阻塞讀取文件,不會放入事件隊列 } catch (error) { console.log(" [*]can't found file! "); throw error; } // console.log(code); // 3、執(zhí)行代碼,所要執(zhí)行的代碼,需要營造一個獨立的空間 // 定義一個數(shù)據(jù)容器,用容器去裝模塊導(dǎo)出的成員 let _module = { // 每個js模塊都會有一個全局的module變量 id:'.', exports:{}, parent:module, filename:filename }; let _exports = _module.exports; // 營造一個獨立空間 code = `(function($require,module,_exports,__dirname,__filename){ $[code] })($require,_module,_exports,dirname,filename)`; // 執(zhí)行代碼 eval(code); // 緩存 $require.cache[filename] = _module; // 返回結(jié)果 return _module.exports; } setInterval(()=>{ const rr = $require("./test.js"); console.log(rr); },1000);
上面的模塊測試使用的兩個模塊
//test.js const date = $require('./date.js'); console.log(module); module.exports = date; //date.js module.exports = new Date();
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作能帶來一定的幫助,如果有疑問大家可以留言交流。
相關(guān)文章
nodejs做個爬蟲爬取騰訊動漫內(nèi)容簡單實現(xiàn)
這篇文章主要為大家介紹了nodejs做個爬蟲爬取騰訊動漫內(nèi)容簡單實現(xiàn),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-07-07npm?install安裝失敗報錯:The?operation?was?rejected?by?your?
這篇文章主要給大家介紹了關(guān)于npm?install安裝失敗報錯:The?operation?was?rejected?by?your?operating?system的相關(guān)資料,文中給出了多種解決方法供大家參考學(xué)習(xí),需要的朋友可以參考下2023-04-04NodeJS?Express使用ORM模型訪問關(guān)系型數(shù)據(jù)庫流程詳解
這篇文章主要介紹了NodeJS?Express使用ORM模型訪問關(guān)系型數(shù)據(jù)庫流程,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)吧2023-01-01nodejs部署到騰訊云服務(wù)器的實現(xiàn)(寶塔面板linux系統(tǒng))
本文主要介紹了nodejs部署到騰訊云服務(wù)器的實現(xiàn)(寶塔面板linux系統(tǒng)),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06node.js連接MongoDB數(shù)據(jù)庫的2種方法教程
這幾天一直在學(xué)習(xí)mongdb的基礎(chǔ)知識,跟著網(wǎng)上大神的腳步(代碼)去模擬連接mongodb數(shù)據(jù)庫,下面這篇文章就給大家總結(jié)介紹了node.js連接MongoDB數(shù)據(jù)庫的2種方法教程,文中介紹的非常詳細,需要的朋友可以參考借鑒,下面來一起看看吧。2017-05-05用node-webkit把web應(yīng)用打包成桌面應(yīng)用(windows環(huán)境)
這篇文章主要介紹了windows環(huán)境下用node-webkit把web應(yīng)用打包成桌面應(yīng)用的教程,需要的朋友可以參考下2018-02-02