摘要:本文重點是講解如何解決循環依賴這個問題。如何找到循環依賴的的文件上文的示例代碼很簡單,個文件,很容易找出循環依賴。如果有十幾個文件,手工去找循環依賴的文件,也是非常麻煩的。
本文重點是講解如何解決循環依賴這個問題。關心這個問題是如何產生的,可以自行谷歌。
如何重現這個問題// a.js const {sayB} = require("./b.js") sayB() function sayA () { console.log("say A") } module.exports = { sayA }
// b.js const {sayA} = require("./a.js") sayA() function sayB () { console.log("say B") } module.exports = { sayB }
執行下面的代碼
? test git:(master) ? node a.js /Users/dd/wj-gitlab/tools/test/b.js:3 sayA() ^ TypeError: sayA is not a function at Object.(/Users/dd/wj-gitlab/tools/test/b.js:3:1) at Module._compile (module.js:635:30) at Object.Module._extensions..js (module.js:646:10) at Module.load (module.js:554:32) at tryModuleLoad (module.js:497:12) at Function.Module._load (module.js:489:3) at Module.require (module.js:579:17) at require (internal/module.js:11:18) at Object. (/Users/dd/wj-gitlab/tools/test/a.js:1:78) at Module._compile (module.js:635:30)
sayA is not a function那么sayA是個什么呢,實際上它是 undefined
遇到這種問題時,你最好能意識到可能是循環依賴的問題,否則找問題可能事倍功半。
如何找到循環依賴的的文件上文的示例代碼很簡單,2個文件,很容易找出循環依賴。如果有十幾個文件,手工去找循環依賴的文件,也是非常麻煩的。
下面推薦一個工具 madge, 它可以可視化的查看文件之間的依賴關系。
注意下圖1,以cli.js為起點,所有的箭頭都是向右展開的,這說明沒有循環依賴。如果有箭頭出現向左逆流,那么就可能是循環依賴的點。
圖2中,出現向左的箭頭,說明出現了循環依賴,說明要此處斷開循環。
【圖1】
【圖2】
將module.exports放到文件頭部,先將自身模塊導出,然后再導入其他模塊。
來自:http://maples7.com/2016/08/17...
// a.js module.exports = { sayA } const {sayB} = require("./b.js") sayB() function sayA () { console.log("say A") }
// b.js module.exports = { sayB } const {sayA} = require("./a.js") console.log(typeof sayA) sayA() function sayB () { console.log("say A") }方案2: 間接調用
通過引入一個event的消息傳遞,讓多個個模塊可以間接傳遞消息,多個模塊之間也可以通過發消息相互調用。
// a.js require("./b.js") const bus = require("./bus.js") bus.on("sayA", sayA) setTimeout(() => { bus.emit("sayB") }, 0) function sayA () { console.log("say A") } module.exports = { sayA }
// b.js const bus = require("./bus.js") bus.on("sayB", sayB) setTimeout(() => { bus.emit("sayA") }, 0) function sayB () { console.log("say B") } module.exports = { sayB }
// bus.js const EventEmitter = require("events") class MyEmitter extends EventEmitter {} module.exports = new MyEmitter()總結
出現循環依賴,往往是代碼的結構出現了問題。應當主動去避免循環依賴這種問題,但是遇到這種問題,無法避免時,也要意識到是循環依賴導致的問題,并找方案解決。
最后給出一個有意思的問題,下面的代碼運行node a.js會輸出什么?為什么會這樣?
// a.js var moduleB = require("./b.js") setInterval(() => { console.log("setInterval A") }, 500) setTimeout(() => { console.log("setTimeout moduleA") moduleB.sayB() }, 2000) function sayA () { console.log("say A") } module.exports = { sayA }
// b.js var moduleA = require("./a.js") setInterval(() => { console.log("setInterval B") }, 500) setTimeout(() => { console.log("setTimeout moduleB") moduleA.sayA() }, 2000) function sayB () { console.log("say B") } module.exports = { sayB }
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/98719.html
摘要:前言最近參加了幾場面試,積累了一些高頻面試題,我把面試題分為兩類,一種是基礎試題主要考察前端技基礎是否扎實,是否能夠將前端知識體系串聯。 前言 最近參加了幾場面試,積累了一些高頻面試題,我把面試題分為兩類,一種是基礎試題: 主要考察前端技基礎是否扎實,是否能夠將前端知識體系串聯。一種是開放式問題: 考察業務積累,是否有自己的思考,思考問題的方式,這類問題沒有標準答案。 基礎題 題目的答...
摘要:回調函數是在異步操作完成后傳播其操作結果的函數,總是用來替代同步操作的返回指令。下面的圖片顯示了中事件循環過程當異步操作完成時,執行權就會交給這個異步操作開始的地方,即回調函數。 本系列文章為《Node.js Design Patterns Second Edition》的原文翻譯和讀書筆記,在GitHub連載更新,同步翻譯版鏈接。 歡迎關注我的專欄,之后的博文將在專欄同步: Enc...
摘要:問題什么是調用棧并且它是的一部分么調用棧當然是的一部分。為什么理解是重要的因為你在每個進程中只能獲取一個調用棧。它是一個從事件隊列中跳去事件的循環并且將它們的回調壓入到調用棧中。當調用棧為空的時候,事件循環可以決定下一步執行哪一個。 你并不知道Node 原文:You don’t know Node 譯者:neal1991 welcome to star my articles-tra...
摘要:深入淺出一直想致力于寫一篇關于廣義講解系統的文章,苦于時間有限,資源有限。事件驅動機制是通過內部單線程高效率地維護事件循環隊列來實現的,沒有多線程的資源占用和上下文的切換。 深入淺出Node.js 一直想致力于寫一篇關于廣義講解Node.js系統的文章,苦于時間有限,資源有限。這篇文章是在結合自己的學習心得以及與行業大佬共同探討下爭對于熟練掌握JS語言后的廣義Node.js.至于為什么...
摘要:深入淺出一直想致力于寫一篇關于廣義講解系統的文章,苦于時間有限,資源有限。事件驅動機制是通過內部單線程高效率地維護事件循環隊列來實現的,沒有多線程的資源占用和上下文的切換。 深入淺出Node.js 一直想致力于寫一篇關于廣義講解Node.js系統的文章,苦于時間有限,資源有限。這篇文章是在結合自己的學習心得以及與行業大佬共同探討下爭對于熟練掌握JS語言后的廣義Node.js.至于為什么...
閱讀 1842·2021-09-22 15:23
閱讀 3274·2021-09-04 16:45
閱讀 1886·2021-07-29 14:49
閱讀 2774·2019-08-30 15:44
閱讀 1527·2019-08-29 16:36
閱讀 1045·2019-08-29 11:03
閱讀 1512·2019-08-26 13:53
閱讀 513·2019-08-26 11:57