Best way to handle decoupling in Go with similar structs in two different packages but subitem in struct making it difficult?

huangapple go评论104阅读模式
英文:

Best way to handle decoupling in Go with similar structs in two different packages but subitem in struct making it difficult?

问题

我对Go相对较新,一直在进行大规模重写,试图尽量减少依赖关系图。我对我所做的工作感到非常满意,但有一个部分我不确定如何处理最好。如果答案是“你将在这两者之间有依赖关系”,那也没关系,我只是在寻找一个好的方法,不期望奇迹发生。

所以下面我有两个包,ab,它们都有相同的结构体。通常情况下,你可以在主函数中将一个转换为另一个,但每个结构体都有一个子项,该子项也是一个结构体,这会阻止Go允许这样做,即使子项具有相同的签名。

一种方法是在B中的结构体中引用A.TzConfig,并允许存在依赖关系,但这正是我想要摆脱的。

我猜另一种方法是创建一个接口,然后通过方法获取Loc的值,我认为这样可以工作,但我还没有尝试过,因为这意味着为仅仅是数据结构的东西创建方法(实际结构体有很多项,我在这里简化了它们),这似乎有点过度设计。

我可以将TzConfig移动到第三个模块中,这样它们两个都会引用那个模块,而不是一个引用另一个,这是我想到的全部方法。

所以我的问题是,对于有实际经验的人来说,在Go中处理这种情况的最佳方法是什么?

我应该提到它们具有重复的结构体的原因只是因为我试图打破它们之间的依赖关系,原始代码只在一个包中有结构体,而另一个包引用它。

  1. package a
  2. type Cfg struct {
  3. Addr string
  4. Loc TzConfig
  5. }
  6. type TzConfig struct {
  7. String string
  8. TZ *time.Location `validate:"noDescent"`
  9. }
  10. func GetCfg() Cfg {
  11. t, _ := time.LoadLocation("MST")
  12. return Cfg{
  13. Addr: "abc",
  14. Host: "a.bc.d",
  15. Loc: config.TzConfig{
  16. String: "MST",
  17. TZ: t,
  18. },
  19. }
  20. }
  1. package b
  2. type Cfg struct {
  3. Addr string
  4. Loc TzConfig
  5. }
  6. type TzConfig struct {
  7. String string
  8. TZ *time.Location `validate:"noDescent"`
  9. }
  10. func DoSomethingWithConfig(c Cfg) {
  11. fmt.Println(c)
  12. }
  1. package main
  2. main() {
  3. c := a.GetCfg()
  4. d := b.DoSomethingWithConfig(b.Cg(c))
  5. fmt.Println(d)
  6. }
英文:

I'm relatively new to go and have been doing a massive rewrite trying to reduce my dependency graph as much as possible. I'm pretty happy with where I got it but there is this one part I am not sure how to best handle. If the answer is, "You are going to have that dependency between the two", that's fine too, I'm just looking for a good approach, not expecting miracles.

So below I have two packages, a & b, and they both have identical structs. Normally you could convert one to the other in main but each has a subitem which is also a struct and that's stopping Go from allowing it even though the subitems have identical signatures.

One way would be to just reference A.TzConfig in the struct in B and let there be a dependency but that's what I am trying to get rid of.

I guess another way is to create an interface and then get the values of Loc through methods, I think that would work but I haven't tried it yet as that means creating methods for something that's just a structure of data (the actual structure has a lot of items, I reduced it to the essentials for simplicity here) which seems like overkill.

I could move TzConfig into a third module so they would both reference that instead of one referencing the other and that's about all I have thought of.

So my question is, from someone with a real experience, what would be the best way to deal with this scenario in go?

I should mention that the reason they have duplicated structs was just because I was trying to break the dependency between them, the original code just had the struct in one package and the other package referencing it.

  1. package a
  2. type Cfg struct {
  3. Addr string
  4. Loc TzConfig
  5. }
  6. type TzConfig struct {
  7. String string
  8. TZ *time.Location `validate:"noDescent"`
  9. }
  10. func GetCfg() Cfg {
  11. t, _ := time.LoadLocation(`MST`)
  12. return Cfg{
  13. Addr: "abc",
  14. Host: "a.bc.d",
  15. Loc: config.TzConfig{
  16. String: "MST",
  17. TZ: t,
  18. },
  19. }
  20. }
  1. package b
  2. type Cfg struct {
  3. Addr string
  4. Loc TzConfig
  5. }
  6. type TzConfig struct {
  7. String string
  8. TZ *time.Location `validate:"noDescent"`
  9. }
  10. func DoSomethingWithConfig(c Cfg) {
  11. fmt.Println(c)
  12. }
  1. package main
  2. main() {
  3. c := a.GetCfg()
  4. d := b.DoSomethingWithConfig(b.Cg(c))
  5. fmt.Println(d)
  6. }

答案1

得分: 0

IMHO,@BurakSerdar提供的建议完全可以,并且非常适合你的情况。我已经按照以下方式重新编写了代码。

common

  1. package common
  2. import "time"
  3. type Cfg struct {
  4. Addr string
  5. Loc TzConfig
  6. }
  7. type TzConfig struct {
  8. String string
  9. TZ *time.Location `validate:"noDescent"`
  10. }

在这里,你应该放置共享的结构体、函数、方法等。

a

  1. package a
  2. import (
  3. "dependencies/common"
  4. "time"
  5. )
  6. type Cfg struct {
  7. common.Cfg
  8. Host string
  9. }
  10. func GetCfg() Cfg {
  11. t, _ := time.LoadLocation("MST")
  12. return Cfg{
  13. Cfg: common.Cfg{
  14. Addr: "abc",
  15. Loc: common.TzConfig{
  16. String: "MST",
  17. TZ: t,
  18. },
  19. },
  20. Host: "a.bc.d",
  21. }
  22. }

在这里,你可以看到 a 包中的特定代码继承了 common 包中的共享代码,如 import 部分所示。

请注意,我使用了结构体嵌入的特性来获取在 common 包中定义的共享字段。

b

  1. package b
  2. import (
  3. "dependencies/common"
  4. "fmt"
  5. )
  6. func DoSomethingWithConfig(c common.Cfg) string {
  7. return fmt.Sprint(c)
  8. }

在这里,没有什么特别需要提到的。

main

  1. package main
  2. import (
  3. "dependencies/a"
  4. "dependencies/b"
  5. "fmt"
  6. )
  7. func main() {
  8. c := a.GetCfg()
  9. d := b.DoSomethingWithConfig(c.Cfg)
  10. fmt.Println(d)
  11. }

在这里,代码应该非常直观。我导入了 ab 包以利用它们的功能。

再次强调,这是一个主观的话题,因此没有一种万能的解决方案。对我来说,这看起来很整洁清晰。我肯定会选择这种方式。请告诉我你的想法,谢谢!

英文:

IMHO, the suggestion provided by @BurakSerdar are completely fine and fits very well for your scenario. I've rewritten the code in this way.

package common

  1. package common
  2. import "time"
  3. type Cfg struct {
  4. Addr string
  5. Loc TzConfig
  6. }
  7. type TzConfig struct {
  8. String string
  9. TZ *time.Location `validate:"noDescent"`
  10. }

Here, you should put the common structs, functions, methods, and so on.

package a

  1. package a
  2. import (
  3. "dependencies/common"
  4. "time"
  5. )
  6. type Cfg struct {
  7. common.Cfg
  8. Host string
  9. }
  10. func GetCfg() Cfg {
  11. t, _ := time.LoadLocation(`MST`)
  12. return Cfg{
  13. Cfg: common.Cfg{
  14. Addr: "abc",
  15. Loc: common.TzConfig{
  16. String: "MST",
  17. TZ: t,
  18. },
  19. },
  20. Host: "a.bc.d",
  21. }
  22. }

Here, you have the specific code related to the package a that inherits the shared code from the common package, as you can see in the import section.
> Please note that I used the structs embedding feature to get the shared fields defined within the common package.

package b

  1. package b
  2. import (
  3. "dependencies/common"
  4. "fmt"
  5. )
  6. func DoSomethingWithConfig(c common.Cfg) string {
  7. return fmt.Sprint(c)
  8. }

Here, there is nothing special to mention.

package main

  1. package main
  2. import (
  3. "dependencies/a"
  4. "dependencies/b"
  5. "fmt"
  6. )
  7. func main() {
  8. c := a.GetCfg()
  9. d := b.DoSomethingWithConfig(c.Cfg)
  10. fmt.Println(d)
  11. }

Here, the code should be pretty straightforward. I imported packages a and b to exploit their functionalities.

Again, I'd like to be clear that this is a subjective topic, hence there isn't a silver bullet solution. To me, looks neat and clear. I would have chosen this way for sure. Let me know and thanks!

huangapple
  • 本文由 发表于 2023年6月18日 19:32:45
  • 转载请务必保留本文链接:https://go.coder-hub.com/76500318.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定