国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > Golang > 正文

Golang報“import cycle not allowed”錯誤的2種解決方法

2020-04-01 18:54:55
字體:
供稿:網(wǎng)友

前言

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

1. 應(yīng)用場景

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

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

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

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

2. 代碼實現(xiàn)

其程序大致如下:

package a 代碼如下:

package aimport ( "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 bimport ( "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 cimport "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,同時 package b 依賴 package a 、 package c 也依賴 package a 。

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

package mainimport "github.com/ggq89/mutualdep/a"func main() { a := a.New(3) a.Pb.DisplayC()}

編譯時就會報錯如下:

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 中補充缺少的方法;

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

package bimport ( "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 中補充可能缺少的方法;

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

package aimport ( "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)在是另一個相互依賴問題:

A depends on C 
C depends on A

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

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

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

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

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

方法移動到 package f 后, package a 的代碼如下:

package aimport ( "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 cimport ( "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ù)導致的相互依賴只能通過分包解決;分包能細化包的功能;

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

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對VEVB武林網(wǎng)的支持。


發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
主站蜘蛛池模板: 普兰店市| 游戏| 巴彦县| 梅河口市| 浪卡子县| 稷山县| 江川县| 漳平市| 报价| 福建省| 五华县| 哈巴河县| 开鲁县| 新巴尔虎右旗| 仙桃市| 灯塔市| 望江县| 庐江县| 沛县| 德保县| 九台市| 廊坊市| 淮安市| 霍林郭勒市| 五常市| 松滋市| 石泉县| 和平县| 阳春市| 天气| 建宁县| 深州市| 新民市| 聂拉木县| 安新县| 肃宁县| 盘山县| 宝山区| 岳阳市| 德州市| 易门县|