Go语言工厂模式自动注册——管理多个包的结构体

本例利用包的 init 特性,将 cls1 和 cls2 两个包注册到工厂,使用字符串创建这两个注册好的结构实例。

完整代码的结构如下:

.
└── src
    └── chapter08
        └── clsfactory
            ├── main.go
            └── base
                └── factory.go
            └── cls1
                └── reg.go
            └── cls2
                └── reg.go

本套教程所有源码下载地址: https://pan.baidu.com/s/1ORFVTOLEYYqDhRzeq0zIiQ    提取密码:hfyf

类工厂(具体文件:…/chapter08/clsfactory/base/factory.go)

 
  1. package base
  2. // 类接口
  3. type Class interface {
  4. Do()
  5. }
  6. var (
  7. // 保存注册好的工厂信息
  8. factoryByName = make(map[string]func() Class)
  9. )
  10. // 注册一个类生成工厂
  11. func Register(name string, factory func() Class) {
  12. factoryByName[name] = factory
  13. }
  14. // 根据名称创建对应的类
  15. func Create(name string) Class {
  16. if f, ok := factoryByName[name]; ok {
  17. return f()
  18. } else {
  19. panic("name not found")
  20. }
  21. }

这个包叫base,负责处理注册和使用工厂的基础代码,该包不会引用任何外部的包。

以下是对代码的说明:

  • 第 4 行定义了“产品”:类。
  • 第 10 行使用了一个 map 保存注册的工厂信息。
  • 第 14 行提供给工厂方注册使用,所谓的“工厂”,就是一个定义为func() Class的普通函数,调用此函数,创建一个类实例,实现的工厂内部结构体会实现 Class 接口。
  • 第 19 行定义通过名字创建类实例的函数,该函数会在注册好后调用。
  • 第 20 行在已经注册的信息中查找名字对应的工厂函数,找到后,在第 21 行调用并返回接口。
  • 第 23 行是如果创建的名字没有找到时,报错。


类1及注册代码(具体文件:…/chapter08/clsfactory/cls1/reg.go)

 
  1. package cls1
  2. import (
  3. "chapter08/clsfactory/base"
  4. "fmt"
  5. )
  6. // 定义类1
  7. type Class1 struct {
  8. }
  9. // 实现Class接口
  10. func (c *Class1) Do() {
  11. fmt.Println("Class1")
  12. }
  13. func init() {
  14. // 在启动时注册类1工厂
  15. base.Register("Class1", func() base.Class {
  16. return new(Class1)
  17. })
  18. }

上面的代码展示了Class1的工厂及产品定义过程。

  • 第 9~15 行定义 Class1 结构,该结构实现了 base 中的 Class 接口。
  • 第 20 行,Class1 结构的实例化过程叫 Class1 的工厂,使用 base.Register() 函数在 init() 函数被调用时与一个字符串关联,这样,方便以后通过名字重新调用该函数并创建实例。


类2及注册代码(具体文件:…/chapter08/clsfactory/cls2/reg.go)

 
  1. package cls2
  2. import (
  3. "chapter08/clsfactory/base"
  4. "fmt"
  5. )
  6. // 定义类2
  7. type Class2 struct {
  8. }
  9. // 实现Class接口
  10. func (c *Class2) Do() {
  11. fmt.Println("Class2")
  12. }
  13. func init() {
  14. // 在启动时注册类2工厂
  15. base.Register("Class2", func() base.Class {
  16. return new(Class2)
  17. })
  18. }

Class2 的注册与 Class1 的定义和注册过程类似。

类工程主流程(具体文件:…/chapter08/clsfactory/main.go)

 
  1. package main
  2. import (
  3. "chapter08/clsfactory/base"
  4. _ "chapter08/clsfactory/cls1" // 匿名引用cls1包, 自动注册
  5. _ "chapter08/clsfactory/cls2" // 匿名引用cls2包, 自动注册
  6. )
  7. func main() {
  8. // 根据字符串动态创建一个Class1实例
  9. c1 := base.Create("Class1")
  10. c1.Do()
  11. // 根据字符串动态创建一个Class2实例
  12. c2 := base.Create("Class2")
  13. c2.Do()
  14. }

下面是对代码的说明:

  • 第 5 和第 6 行使用匿名引用方法导入了 cls1 和 cls2 两个包。在 main() 函数调用前,这两个包的 init() 函数会被自动调用,从而自动注册 Class1 和 Class2。
  • 第 12 和第 16 行,通过 base.Create() 方法查找字符串对应的类注册信息,调用工厂方法进行实例创建。
  • 第 13 和第 17 行,调用类的方法。


执行下面的指令进行编译:

export GOPATH=/home/davy/golangbook/code
go install chapter08/clsfactory
$GOPATH/bin/clsfactory

代码输出如下:

Class1
Class2

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值