學(xué)習(xí) Go 語(yǔ)言的開(kāi)發(fā)者越來(lái)越多了,很多小伙伴在使用時(shí),就會(huì)遇到種種不理解的問(wèn)題。
其中一點(diǎn)就是包的循環(huán)引用的報(bào)錯(cuò):
1
2
3
4
|
package command -line-arguments imports github.com /eddycjy/awesome-project/a imports github.com /eddycjy/awesome-project/b imports github.com /eddycjy/awesome-project/a : import cycle not allowed |
一下子就很懵逼了,為什么 Go 不支持包之間的循環(huán)引用呢,這就很不解了,難道還影響性能了?
如下圖:
今天煎魚(yú)將和大家一起了解背后的原因。
1、案例演示
這里我們做一個(gè)基本的案例 Demo
,便于沒(méi)接觸過(guò)的同學(xué)建立初步認(rèn)知。我們的程序分別有 2 個(gè) package
。
package a
的代碼如下:
1
2
3
4
5
6
7
|
import ( "github.com/eddycjy/awesome-project/b" ) func Hello(s string) { b.Print(s) } |
package b 的代碼如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
import ( "fmt" "github.com/eddycjy/awesome-project/a" ) func Hello() { a.Hello( "腦子進(jìn)煎魚(yú)了" ) } func Print(s string) { fmt .Println(s) } |
再在 main.go
的文件中調(diào)用 a.Hello
("腦子進(jìn)煎魚(yú)了") 方法。
一運(yùn)行,就會(huì)出現(xiàn)如下錯(cuò)誤提示:
package command-line-arguments
imports github.com/eddycjy/awesome-project/a
imports github.com/eddycjy/awesome-project/b
imports github.com/eddycjy/awesome-project/a: import cycle not allowed
錯(cuò)誤的本質(zhì)原因是 package a
引用了 package b
,而 package b
又引用了 package a
,造成了循環(huán)引用。
這在 Go 語(yǔ)言中是明令禁止的,在編譯時(shí)就會(huì)中斷程序,導(dǎo)致編譯失敗。
2、原因分析
根據(jù)現(xiàn)在 Go 官方的統(tǒng)一意見(jiàn)來(lái)看,package
循環(huán)導(dǎo)入幾乎不可能出現(xiàn),即使是 Go2,也被明確拒絕了。
因?yàn)?Go2 可能是很多核心問(wèn)題的破變的關(guān)鍵節(jié)點(diǎn),有許多人提了類(lèi)似《proposal: Go 2: allow import cycle》的提案,希望解決循環(huán)引入的問(wèn)題。
Go 語(yǔ)言之父 Rob Pike
親自回答了這個(gè)問(wèn)題,原因如下:
沒(méi)有支持循環(huán)引用:目的是迫使 Go 程序員更多地考慮程序的依賴(lài)關(guān)系。
- 保持依賴(lài)關(guān)系圖的簡(jiǎn)潔。
- 快速的程序構(gòu)建。
如果支持循環(huán)引用:很容易會(huì)造成懶惰、不良的依賴(lài)性管理和緩慢的構(gòu)建。這是設(shè)計(jì)者不希望看見(jiàn)的。
- 混亂的依賴(lài)關(guān)系。
- 緩慢的程序構(gòu)建
因此考慮一開(kāi)始就保持依賴(lài)圖的正確 DAG
,Rob Pike
認(rèn)為這是一個(gè)值得預(yù)先簡(jiǎn)化的領(lǐng)域。
在 Go 程序中去做導(dǎo)入循環(huán)這件事可能很方便,但背后的代價(jià)可能是災(zāi)難性的,會(huì)對(duì) Go 的構(gòu)建性能和依賴(lài)關(guān)系造成非常不利的影響。
所以在 Go 中被明確禁止支持。
3、總結(jié)
在程序中,如果我們頻繁的出現(xiàn)模塊與模塊之間的循環(huán)引用,這時(shí)候我們是不是應(yīng)該考慮一下,是不是設(shè)計(jì)的有些問(wèn)題,要不要考慮調(diào)整?
但也并非所有的事都是二極管,Go 源碼可能或多或少都有自己循環(huán)引用的案例,最重要的是想清楚。
到此這篇關(guān)于為什么GO不支持循環(huán)引用的文章就介紹到這了,更多相關(guān)GO不支持循環(huán)引用內(nèi)容請(qǐng)搜索服務(wù)器之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持服務(wù)器之家!
原文鏈接:https://www.tuicool.com/articles/eeYF3um