<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          為什么 Go 不支持循環(huán)引用?

          共 1838字,需瀏覽 4分鐘

           ·

          2021-11-17 23:15

          大家好,我是煎魚(yú)。

          學(xué)習(xí) Go 語(yǔ)言的開(kāi)發(fā)者越來(lái)越多了,很多小伙伴在使用時(shí),就會(huì)遇到種種不理解的問(wèn)題。

          其中一點(diǎn)就是包的循環(huán)引用的報(bào)錯(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

          一下子就很懵逼了,為什么 Go 不支持包之間的循環(huán)引用呢,這就很不解了,難道還影響性能了?

          如下圖:

          圖來(lái)自網(wǎng)絡(luò)

          今天煎魚(yú)將和大家一起了解背后的原因。

          案例演示

          這里我們做一個(gè)基本的案例 Demo,便于沒(méi)接觸過(guò)的同學(xué)建立初步認(rèn)知。我們的程序分別有 2 個(gè) package。

          package a 的代碼如下:

          import?(
          ?"github.com/eddycjy/awesome-project/b"
          )

          func?Hello(s?string)?{
          ?b.Print(s)
          }

          package b 的代碼如下:

          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)致編譯失敗。

          原因分析

          根據(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),有許多人提了類似《proposal: Go 2: allow import cycle》的提案,希望解決循環(huán)引入的問(wèn)題。

          Go 語(yǔ)言之父 Rob Pike 親自回答了這個(gè)問(wèn)題,原因如下:

          • 沒(méi)有支持循環(huán)引用:目的是迫使 Go 程序員更多地考慮程序的依賴關(guān)系。
            • 保持依賴關(guān)系圖的簡(jiǎn)潔。
            • 快速的程序構(gòu)建。
          • 如果支持循環(huán)引用:很容易會(huì)造成懶惰、不良的依賴性管理和緩慢的構(gòu)建。這是設(shè)計(jì)者不希望看見(jiàn)的。
            • 混亂的依賴關(guān)系。
            • 緩慢的程序構(gòu)建

          因此考慮一開(kā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)建性能和依賴關(guān)系造成非常不利的影響

          所以在 Go 中被明確禁止支持。

          總結(jié)

          在程序中,如果我們頻繁的出現(xiàn)模塊與模塊之間的循環(huán)引用,這時(shí)候我們是不是應(yīng)該考慮一下,是不是設(shè)計(jì)的有些問(wèn)題,要不要考慮調(diào)整?

          但也并非所有的事都是二極管,Go 源碼可能或多或少都有自己循環(huán)引用的案例,最重要的是想清楚。

          你對(duì)此支持循環(huán)引用怎么看,歡迎在評(píng)論區(qū)留言交流:)

          關(guān)注煎魚(yú),吸取他的知識(shí)???



          你好,我是煎魚(yú)。高一折騰過(guò)前端,參加過(guò)國(guó)賽拿了獎(jiǎng),大學(xué)搞過(guò) PHP。現(xiàn)在整 Go,在公司負(fù)責(zé)微服務(wù)架構(gòu)等相關(guān)工作推進(jìn)和研發(fā)。

          從大學(xué)開(kāi)始靠自己賺生活費(fèi)和學(xué)費(fèi),到出版 Go 暢銷書(shū)《Go 語(yǔ)言編程之旅》,再到獲得 GOP(Go 領(lǐng)域最有觀點(diǎn)專家)榮譽(yù),點(diǎn)擊藍(lán)字查看我的出書(shū)之路

          日常分享高質(zhì)量文章,輸出 Go 面試、工作經(jīng)驗(yàn)、架構(gòu)設(shè)計(jì),加微信拉讀者交流群,記得點(diǎn)贊!

          瀏覽 80
          點(diǎn)贊
          評(píng)論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          評(píng)論
          圖片
          表情
          推薦
          點(diǎn)贊
          評(píng)論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  婷婷五月天97 | 亚洲A片在线观看 | 免费看黄色做爱视频 | 麻豆91麻豆国产传媒 | 亚洲成人网站免费在线观看 |