萬盛學電腦網

 萬盛學電腦網 >> 腳本專題 >> javascript >> 淺析Node.js中使用依賴注入的相關問題及解決方法

淺析Node.js中使用依賴注入的相關問題及解決方法

  這篇文章主要介紹了淺析Node.js中使用依賴注入的相關問題及解決方法,Node.js是一個將JavaScript應用運行於服務器端的框架,需要的朋友可以參考下

  最近,我轉向使用依賴注入來幫助理解分離代碼的簡單途徑,並有助測試。然而,Node.js中的模塊依賴Node提供的系統API,這很難判斷私有依賴被恰當的使用。一般的依賴注入很難在這種情況下使用,但現在不要放棄希望。

  requireCauses 問題

  Node.js很容易依照需求導入依賴。它運行的很好,並且比AMD模式加載器例如RequireJS要簡單。當我們模擬那些依賴的時候問題就來了。如果Node.js中模型的加載是受控的,我們怎麼做才能控制讓偽對象在測試期間被使用到?我們可以使用Node的vm模式,通過vm我們可以再新的上下文中加載模型。運行在新的上下文中,我們可以控制需求反射出模型的方法。

  解決方案

  謝謝這篇文章, 現在可以給你提供一個相當不錯的解決方案. 代碼在下面:

  ?

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 var vm = require('vm'); var fs = require('fs'); var path = require('path');   /** * Helper for unit testing: * – load module with mocked dependencies * – allow accessing private state of the module * * @param {string} filePath Absolute path to module (file to load) * @param {Object=} mocks Hash of mocked dependencies */ exports.loadModule = function(filePath, mocks) { mocks = mocks || {};   // this is necessary to allow relative path modules within loaded file // i.e. requiring ./some inside file /a/b.js needs to be resolved to /a/some var resolveModule = function(module) { if (module.charAt(0) !== '.') return module; return path.resolve(path.dirname(filePath), module); };   var exports = {}; var context = { require: function(name) { return mocks[name] || require(resolveModule(name)); }, console: console, exports: exports, module: { exports: exports } };   vm.runInNewContext(fs.readFileSync(filePath), context); return context; };

  你也可以在 這裡 下載代碼片段 . 雖然在不是在文章發布最多的代碼, 他仍然可以使用一些解釋. 當我們測試時, 我們要加載這個模塊進入測試, 使用theloadModulefunction代替ofrequire加載模塊測試.

  第一個參數filePath指定了我們要測試模型的查找位置。第二個參數mocks包含一個對象,對象的屬性名稱要和我們嘗試require的模型的名稱相匹配。那些屬性指定的值就是偽對象,用來代替一般被require的模型。

  本質上看就是用vm來加載和運行模型在另一個“上下文”中。換句話說,我們重建了全局變量(例如require和exports)以便我們能控制它們。需要注意的是我們編寫了一個可用的新require函數。所做一切就是檢查一下用執行的名字是否有一個模擬的依賴,如果每日有,我就就把它委托給那個常用的require函數。

  使用模塊加載器的例子

  如果你還有點困惑,你可以看下面的代碼示例,看它在上下文中的使用,也許能幫你清楚一些。首先,我們創建一個簡單的模塊。

  ?

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 var fs = require('fs');   module.exports = { // Do something with `fs` } 想象一下這個很酷,對嗎?不管怎樣,現在我們測試那個模塊,但是我們要模擬fs來看看它是怎麼在內部使用的。   // Jasmine's syntax http://pivotal.github.com/jasmine/ describe('someModule', function() { var loadModule = require('module-loader').loadModule; var module, fsMock;   beforeEach(function() { fsMock = { // a mock for `fs` };   // load the module with mock fs instead of real fs module = loadModule('./web-server.js', {fs: fsMock}); });   it('should work', function() { // a test that utilizes the fact that we can now control `fs` }); });

 

  主要注意是在7至12行,我們為fs創建了一個偽對象並使用我們新的loadModule函數將這個使用的對象聯系到上面的小模塊中(我的意思是真棒!請記住,這是真棒,對不對?)。

copyright © 萬盛學電腦網 all rights reserved