濮阳杆衣贸易有限公司

主頁 > 知識(shí)庫 > Golang報(bào)“import cycle not allowed”錯(cuò)誤的2種解決方法

Golang報(bào)“import cycle not allowed”錯(cuò)誤的2種解決方法

熱門標(biāo)簽:浙江高速公路地圖標(biāo)注 地圖標(biāo)注的汽車標(biāo) 廣州呼叫中心外呼系統(tǒng) 西部云谷一期地圖標(biāo)注 學(xué)海導(dǎo)航地圖標(biāo)注 南通如皋申請開通400電話 中國地圖標(biāo)注省會(huì)高清 高德地圖標(biāo)注口訣 江西轉(zhuǎn)化率高的羿智云外呼系統(tǒng)

前言

相信不少 Gopher 在寫 Golang 程序都遇到過 import cycle not allowed 問題,本人最近研讀 go-ethereum 源碼時(shí),發(fā)現(xiàn)定義 interface 也能解決此問題, 還能解決連分包都不能解決的情況, 并且比分包更加簡單快捷。下面逐個(gè)講解 分包 和 定義接口 這兩種方法。

1. 應(yīng)用場景

假設(shè)有如下使用場景:

A 是應(yīng)用程序的框架級(jí)結(jié)構(gòu)體,在 A 包含子模塊 B 和 C 的指針;

B 為了方便的使用應(yīng)用的其他子模塊(比如 C )功能,所以在其結(jié)構(gòu)體包含了 A 的指針;

C 要調(diào)用 A 包中的某個(gè)方法;

2. 代碼實(shí)現(xiàn)

其程序大致如下:

package a 代碼如下:

package a

import (
 "fmt"

 "github.com/ggq89/mutualdep/b"
 "github.com/ggq89/mutualdep/c"
)

type A struct {
 Pb *b.B
 Pc *c.C
}

func New(ic int) *A {
 a := A{
 Pc: c.New(ic),
 }

 a.Pb = b.New(a)

 return a
}

func Printf(v int) {
 fmt.Printf("%v", v)
}

package b 代碼如下:

package b

import (
 "github.com/ggq89/mutualdep/a"
)

type B struct {
 Pa *a.A
}

func New(a *a.A) *B {
 return B{
 Pa: a,
 }
}

func (b *B) DisplayC() {
 b.Pa.Pc.Show()
}

package c 代碼如下:

package c

import "github.com/ggq89/mutualdep/a"

type C struct {
 Vc int
}

func New(i int) *C {
 return C{
 Vc: i,
 }
}

func (c *C) Show() {
 a.Printf(c.Vc)
}

package a 依賴 package b 和 package c,同時(shí) package b 依賴 package a 、 package c 也依賴 package a 。

main 函數(shù)代碼如下:

package main

import "github.com/ggq89/mutualdep/a"

func main() {
 a := a.New(3)
 a.Pb.DisplayC()
}

編譯時(shí)就會(huì)報(bào)錯(cuò)如下:

import cycle not allowed
package main
    imports github.com/ggq89/mutualdep/a
    imports github.com/ggq89/mutualdep/b
    imports github.com/ggq89/mutualdep/a

3. 定義接口

現(xiàn)在的問題是:

A depends on B
B depends on A

對于 A struct 和 B struct 有彼此的指針這種相互依賴問題,可以使用定義接口的方法解決,具體步驟如下:

在 package b 中 定義 a interface ; 將 b 所有使用到結(jié)構(gòu)體 a 的變量和方法的地方全部轉(zhuǎn)化成 使用接口 a 的方法;在 a interface 中補(bǔ)充缺少的方法;

經(jīng)過上面的步驟處理后, package b 代碼如下:

package b

import (
 "github.com/ggq89/mutualdep/c"
)

type B struct {
 Pa a
}

type a interface {
 GetC() *c.C
}

func New(a a) *B {
 return B{
 Pa:a,
 }
}

func (b *B) DisplayC() {
 b.Pa.GetC().Show()
}

在 package a 中補(bǔ)充可能缺少的方法;

處理后, package a 中的代碼如下:

package a

import (
 "fmt"

 "github.com/ggq89/mutualdep/b"
 "github.com/ggq89/mutualdep/c"
)

type A struct {
 Pb *b.B
 Pc *c.C
}

func New(ic int) *A {
 a := A{
 Pc:c.New(ic),
 }

 a.Pb = b.New(a)

 return a
}

func (a *A)GetC() *c.C {
 return a.Pc
}

func Printf(v int) {
 fmt.Printf("%v", v)
}

4. 拆分包

再次編譯,提示如下:

import cycle not allowed
package main
    imports github.com/ggq89/mutualdep/a
    imports github.com/ggq89/mutualdep/b
    imports github.com/ggq89/mutualdep/c
    imports github.com/ggq89/mutualdep/a

現(xiàn)在是另一個(gè)相互依賴問題:

A depends on C
C depends on A

與前面的相互依賴不同,前面的依賴是由于 A struct 和 B struct 有彼此的指針導(dǎo)致的,屬于硬相互依賴;

而這里是由于 package c 中的方法調(diào)用 package a 中的方法引起的,屬于軟相互依賴;

  • 這種相互依賴可以通過將方法拆分到另一個(gè)包的方式來解決;在拆分包的過程中,可能會(huì)將結(jié)構(gòu)體的方法轉(zhuǎn)化為普通的函數(shù);

引入 package f , 將方法遷移到 f 中 :

package f

import "fmt"

func Printf(v int) {
 fmt.Printf("%v", v)
}

方法移動(dòng)到 package f 后, package a 的代碼如下:

package a

import (
 "github.com/ggq89/mutualdep/b"
 "github.com/ggq89/mutualdep/c"
)

type A struct {
 Pb *b.B
 Pc *c.C
}

func New(ic int) *A {
 a := A{
 Pc: c.New(ic),
 }

 a.Pb = b.New(a)

 return a
}

func (a *A) GetC() *c.C {
 return a.Pc
}

package c隨之改成調(diào)用package f,其代碼如下:

package c

import (
 "github.com/ggq89/mutualdep/a/f"
)

type C struct {
 Vc int
}

func New(i int) *C {
 return C{
 Vc: i,
 }
}

func (c *C) Show() {
 f.Printf(c.Vc)
}

現(xiàn)在依賴關(guān)系如下:

A depends on B and C
B depends on C
C depends on F

至此,兩種包相互依賴關(guān)系都得以解決。

5. 總結(jié)

對于軟相互依賴,利用分包的方法就能解決,有些函數(shù)導(dǎo)致的相互依賴只能通過分包解決;分包能細(xì)化包的功能;

對于硬相互依賴只能通過定義接口的方法解決;定義接口能提高包的獨(dú)立性,同時(shí)也提高了追蹤代碼調(diào)用關(guān)系的難度;

參考文章:

  • golang不允許循環(huán)import問題(“import cycle not allowed”) : https://www.jb51.net/article/145536.htm
  • golang解決import cycle not allowed的一種思路 : https://www.jb51.net/article/145539.htm

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。

您可能感興趣的文章:
  • 對Golang import 導(dǎo)入包語法詳解
  • go各種import的使用方法講解
  • golang 之import和package的使用
  • MongoDB使用mongoexport和mongoimport命令,批量導(dǎo)出和導(dǎo)入JSON數(shù)據(jù)到同一張表的實(shí)例
  • golang中import cycle not allowed解決的一種思路
  • 詳解golang避免循環(huán)import問題(“import cycle not allowed”)
  • 如何解決django配置settings時(shí)遇到Could not import settings ''conf.local''
  • Golang import 導(dǎo)入包語法及一些特殊用法詳解

標(biāo)簽:吐魯番 德宏 貴州 常州 東營 曲靖 許昌 保定

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Golang報(bào)“import cycle not allowed”錯(cuò)誤的2種解決方法》,本文關(guān)鍵詞  Golang,報(bào),import,cycle,not,allowed,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《Golang報(bào)“import cycle not allowed”錯(cuò)誤的2種解決方法》相關(guān)的同類信息!
  • 本頁收集關(guān)于Golang報(bào)“import cycle not allowed”錯(cuò)誤的2種解決方法的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    高阳县| 新泰市| 肥西县| 张家界市| 喀喇沁旗| 长春市| 南宫市| 志丹县| 皋兰县| 象山县| 子长县| 武义县| 峨边| 土默特右旗| 正阳县| 晴隆县| 江川县| 宿州市| 靖西县| 永仁县| 贵港市| 晴隆县| 绥阳县| 陕西省| 岳池县| 镶黄旗| 皋兰县| 中宁县| 泌阳县| 临海市| 磐石市| 娄烦县| 伽师县| 樟树市| 军事| 河南省| 新巴尔虎右旗| 高雄县| 余干县| 秀山| 亚东县|