登录
首页 >  Golang >  Go问答

golang 所用的工厂方法面临循环依赖挑战

来源:stackoverflow

时间:2024-03-09 08:27:27 135浏览 收藏

从现在开始,我们要努力学习啦!今天我给大家带来《golang 所用的工厂方法面临循环依赖挑战》,感兴趣的朋友请继续看下去吧!下文中的内容我们主要会涉及到等等知识点,如果在阅读本文过程中有遇到不清楚的地方,欢迎留言呀!我们一起讨论,一起学习!

问题内容

我意识到有关循环依赖的问题之前已经得到解答,但是,答案通常只是简单地说合并包。长话短说,我有一个许多类型都会实现的接口。我想要一种方法来选择在运行时使用其名称来使用其中一种类型。这也将用于序列化。我读入类的名称,然后实例化正确的类。

我使用了策略模式。 a包中有一个base接口。

package a

import (
    "../c"
)

type base interface {
   dostuff(p c.profile) int
}

type operation struct {
    base base
    data int
}

func (o *operation) execute(p c.profile) int {
    return o.base.dostuff(p)
}

然后,b 包中存在实现该接口的类型。

//file impl1.go
package b

import (
    "../c"
)

type impl1 struct {}

func (b *impl1 ) dostuff(p c.profile) int {
    ...
}

//file impl2.go
package b

import (
    "../c"
)

type impl2 struct {}

func (b *impl2 ) dostuff(p c.profile) int {
    ...
}

然后在包 c 中,我有 struct foo ,其字段类型为 base 。该字段可以指向包 b 中的任何实现。我们可以在运行时选择实现。这是我最终想要序列化的结构。

package C

import (
    "../A"
    "../B"
) 

type Foo struct {
    bar A.Base
    baz []Profile
    ...
}

func (f *Foo) changeBar(name string, data int) {
    switch name {
    case "Impl1":
        f.bar = Operation{Base: B.Impl1{}, Data: data}
    case "Impl2":
        f.bar = Operation{Base: B.Impl2{}, Data: data}
    ...
}

编辑:同样在 c 包中,我们有 profile,这就是 ab 包需要导入它的原因。

此代码具有循环依赖关系 c -> b -> c。一个明显的解决方案是将 profile 移动到不同的包。但这是不可能的,因为 profile (以及 c 包中的许多其他类似类型)和 foo 非常紧密地耦合并且属于同一个包(在这个最小工作示例中可能不那么明显)。这是此类问题的其他答案所建议的,但我想学习如何利用从使用其他语言中学到的良好实践来使其发挥作用。

另一种解决方案是以某种方式将工厂方法 changebar 移动到另一个包,并且仅在 c 包的外部使用它(从而避免循环),将其结果传递给 c 作为参数,但有一些情况(序列化特别是)我实际上需要它在 c 包内部。

我花了很多时间来解决这个问题,但我最终得到的只是要么将所有内容放在一个大包中,要么将每个文件放在一个单独的包中并将所有内容导出。在其他编程语言中,有可能具有这些循环依赖性或一次从包导入一个“类”。 go里面怎么进去?


解决方案


您可以使用类型注册表打破从 c 到 b 的依赖关系:

package c

import "a"

var typeregistry = map[string]func() a.base {}

func (f *foo) changebar(name string, data int) {
   if factory, ok:=typeregistry[name]; ok {
       f.bar=operation{base:factory(),data:data}
   }
}

然后在任何包中注册您的实现:

package B

import "../C"

func init() {
   C.TypeRegistry["myType"]=func() A.Base { return MyType{}}
}

文中关于的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《golang 所用的工厂方法面临循环依赖挑战》文章吧,也可关注golang学习网公众号了解相关技术文章。

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>