golang包循环引用的几种解决方案

LosAngel
• 阅读 2923

golang包循环引用的几种解决方案

发表于2020年11月2日2020年11月3日 作者 libuba

一、前言

golang为了加速编译,不允许包循环引用。通常来说,只要你的包规划得好,严格规范单向调用链(如控制层->业务层->数据层),一般不会出现包循环引用问题。当然现实业务往往不会这么理想,同层级之间的不同包经常需要互相引用,下面我就分享几种解决包循环引用的方案。

二、新建公共接口包(父包),将需要循环调用的函数或方法抽象为接口

  • package_i
package package\_i

type PackageAInterface interface {
    PrintA()
}

type PackageBInterface interface {
    PrintB()
}
  • package_a
package package\_a

import (
    "cycle/package\_i"
    "fmt"
)

type PackageA struct {
    B package\_i.PackageBInterface
}

func (a PackageA) PrintA() {
    fmt.Println("I'm a!")
}

func (a PackageA) PrintAll() {
    a.PrintA()
    a.B.PrintB()
}
  • package_b
package package\_b

import (
    "cycle/package\_i"
    "fmt"
)

type PackageB struct {
    A package\_i.PackageAInterface
}

func (b PackageB) PrintB() {
    fmt.Println("I'm b!")
}

func (b PackageB) PrintAll() {
    b.PrintB()
    b.A.PrintA()
}
  • main
package main

import (
    "cycle/package\_a"
    "cycle/package\_b"
)

func main() {
    a := new(package\_a.PackageA)
    b := new(package\_b.PackageB)
        a.B \= b
        b.A \= a
    a.PrintAll()
    b.PrintAll()
}

三、新建公共组合包(子包),在组合包中组合调用

  • package_c
package package\_c

import (
    "cycle/package\_a"
    "cycle/package\_b"
)

type CombileAB struct {
    A \*package\_a.PackageA
    B \*package\_b.PackageB
}

func (c CombileAB) PrintAll() {
    c.A.PrintA()
    c.B.PrintB()
}
  • main
package main

import (
    "cycle/package\_a"
    "cycle/package\_b"
    "cycle/package\_c"
)

func main() {
    a := new(package\_a.PackageA)
    b := new(package\_b.PackageB)
    c := new(package\_c.CombileAB)
    c.A \= a
    c.B \= b
    c.PrintAll()
}

四、全局存储需要相互依赖的函数,通过关键字进行调用

  • callback_mgr
package callback\_mgr

import (
    "fmt"
    "reflect"
)

var callBackMap map\[string\]interface{}

func init() {
    callBackMap \= make(map\[string\]interface{})
}

func RegisterCallBack(key string, callBack interface{}) {
    callBackMap\[key\] \= callBack
}

func CallBackFunc(key string, args ...interface{}) \[\]interface{} {
    if callBack, ok := callBackMap\[key\]; ok {
        in := make(\[\]reflect.Value, len(args))
        for i, arg := range args {
            in\[i\] \= reflect.ValueOf(arg)
        }
        outList := reflect.ValueOf(callBack).Call(in)
        result := make(\[\]interface{}, len(outList))
        for i, out := range outList {
            result\[i\] \= out.Interface()
        }
        return result
    } else {
        panic(fmt.Errorf("callBack(%s) not found", key))
    }
}
  • package_a
package package\_a

import (
    "cycle/callback\_mgr"
    "fmt"
)

func init() {
    callback\_mgr.RegisterCallBack("getA", new(PackageA).GetA)
}

type PackageA struct {
}

func (a PackageA) GetA() string {
    return "I'm a!"
}

func (a PackageA) PrintAll() {
    fmt.Println(a.GetA())
    fmt.Println(callback\_mgr.CallBackFunc("getB")\[0\].(string))
}
  • package_b
package package\_b

import (
    "cycle/callback\_mgr"
    "fmt"
)

func init() {
    callback\_mgr.RegisterCallBack("getB", new(PackageB).GetB)
}

type PackageB struct {
}

func (b PackageB) GetB() string {
    return "I'm b!"
}

func (b PackageB) PrintAll() {
    fmt.Println(b.GetB())
    fmt.Println(callback\_mgr.CallBackFunc("getA")\[0\].(string))
}
  • main
package main

import (
    "cycle/package\_a"
    "cycle/package\_b"
)

func main() {
    a := new(package\_a.PackageA)
    b := new(package\_b.PackageB)
    a.PrintAll()
    b.PrintAll()
}

五、不需要回调结果的可以通过事件总线(eventBus)解耦

  • eventBus
package eventBus

import (
    "github.com/asaskevich/EventBus"
)

var globalEventBus EventBus.Bus

func init() {
    globalEventBus \= EventBus.New()
}

func Subscribe(topic string, fn interface{}) error {
    return globalEventBus.Subscribe(topic, fn)
}

func SubscribeAsync(topic string, fn interface{}, transactional bool) error {
    return globalEventBus.SubscribeAsync(topic, fn, transactional)
}

func Publish(topic string, args ...interface{}) {
    globalEventBus.Publish(topic, args...)
}
  • package_a
package package\_a

import (
    "cycle/eventBus"
    "fmt"
)

func init() {
    eventBus.Subscribe("PrintA", new(PackageA).PrintA)
}

type PackageA struct {
}

func (a PackageA) PrintA() {
    fmt.Println("I'm a!")
}

func (a PackageA) PrintAll() {
    a.PrintA()
    eventBus.Publish("PrintB")
}
  • package_b
package package\_b

import (
    "cycle/eventBus"
    "fmt"
)

func init() {
    eventBus.Subscribe("PrintB", new(PackageB).PrintB)
}

type PackageB struct {
}

func (b PackageB) PrintB() {
    fmt.Println("I'm b!")
}

func (b PackageB) PrintAll() {
    b.PrintB()
    eventBus.Publish("PrintA")
}

分类: golang随笔

标签: go

本文转自 https://libuba.com/2020/11/02/golang%E5%8C%85%E5%BE%AA%E7%8E%AF%E5%BC%95%E7%94%A8%E7%9A%84%E5%87%A0%E7%A7%8D%E8%A7%A3%E5%86%B3%E6%96%B9%E6%A1%88/,如有侵权,请联系删除。

点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Stella981 Stella981
3年前
Golang Gin实践 番外 请入门 Makefile
<h1GolangGin实践番外请入门Makefile</h1<p原文地址:<ahref"https://github.com/EDDYCJY/blog/blob/master/golang/gin/20180826Gin%E5%AE%9E%E8%B7%B5%E7%95%AA%E5%A4%96%E8%AF%B7%E5%85%A5
Wesley13 Wesley13
3年前
Java面试
<divclass"htmledit\_views"id"content\_views"<pid"maintoc"<strong目录</strong</p<pid"Java%E5%9F%BA%E7%A1%80%EF%BC%9Atoc"style"marginleft:40px;"<ahref"Java%E5%
Stella981 Stella981
3年前
Apollo
现在,各个服务端都启动完毕,我们可以开始研究client端怎么使用apolloclient了。具体请参考https://github.com/ctripcorp/apollo/wiki/Java%E5%AE%A2%E6%88%B7%E7%AB%AF%E4%BD%BF%E7%94%A8%E6%8C%87%E5%8D%971)普通jav
Stella981 Stella981
3年前
Apollo使用文档(Java)
1、apolo的github上的文档很全,大家可以去github上查看下面是java客户端使用文档https://github.com/ctripcorp/apollo/wiki/Java%E5%AE%A2%E6%88%B7%E7%AB%AF%E4%BD%BF%E7%94%A8%E6%8C%87%E5%8D%97(https://www.oschin
Wesley13 Wesley13
3年前
Java 入门进阶
Java入门进阶發表於20150416http://xielong.me/2015/04/16/%E6%9C%8D%E5%8A%A1%E7%AB%AF%E5%B7%A5%E7%A8%8B%E5%B8%88%E5%85%A5%E9%97%A8%E4%B8%8E%E8%BF%9B%E9%98%B6Java%E7%89%88/
Wesley13 Wesley13
3年前
Oracle 数据库勒索病毒 RushQL 处理办法
Oracle数据库勒索病毒RushQL处理办法办法来自Oracle官方:https://blogs.oracle.com/cnsupport\_news/%E5%AF%B9%E6%95%B0%E6%8D%AE%E5%BA%93%E7%9A%84%E2%80%9C%E6%AF%94%E7%89%B9%E5%B8%81%E6%94%BB%E5%8
Stella981 Stella981
3年前
Android蓝牙连接汽车OBD设备
//设备连接public class BluetoothConnect implements Runnable {    private static final UUID CONNECT_UUID  UUID.fromString("0000110100001000800000805F9B34FB");
Wesley13 Wesley13
3年前
Java三大特性
_Java面向对象编程三大特性:封装继承多态(https://snailclimb.gitee.io/javaguide//docs/java/Java%E5%9F%BA%E7%A1%80%E7%9F%A5%E8%AF%86?id_11java%e9%9d%a2%e5%90%91%e5%af%b9%e8%b1%a1%e7%bc%96%e
Wesley13 Wesley13
3年前
2、Libgdx配置你的开发环境(Eclipse,Intellij IDEA,NetBeans)
(原文:http://www.libgdx.cn/topic/13/2libgdx%E9%85%8D%E7%BD%AE%E4%BD%A0%E7%9A%84%E5%BC%80%E5%8F%91%E7%8E%AF%E5%A2%83eclipseintellijideanetbeans(https://www.oschina.net/action/G
LosAngel
LosAngel
Lv1
流浪的月亮和繁密的星辰姗姗来迟。
文章
3
粉丝
0
获赞
0