如何使用Golang实现工厂模式灵活对象创建_Golang工厂方法实例

9次阅读

直接 new struct 不适合多变创建逻辑,因硬编码会导致 if/else 泛滥且新增类型需修改入口;工厂模式将实例化决策剥离,函数型工厂返回接口实现解耦,结构体工厂支持配置与依赖注入,但需注意指针接收者匹配及错误处理。

如何使用 Golang 实现工厂模式灵活对象创建_Golang 工厂方法实例

为什么 直接 new struct 不适合多变的创建逻辑

当业务中需要根据参数动态决定创建哪种具体类型(比如不同支付方式:AlipayWechatPayBankTransfer),硬 编码 &Payment{Type: "alipay"} 会快速导致 if/else 泛滥,且每新增一种类型都要改创建入口。工厂模式把“实例化谁”这个决策从调用方剥离,交给专门的工厂函数或结构体处理。

用函数型工厂实现最简解耦

Go 没有类继承,但函数是一等公民,用返回接口的工厂函数足够轻量。关键点在于:工厂不暴露具体类型,只返回统一接口;调用方完全不知道底层是哪个 struct。

type Payment interface {Process(amount float64) error }  type Alipay struct{} func (a *Alipay) Process(amount float64) error {// 实现     return nil}  type WechatPay struct{} func (w *WechatPay) Process(amount float64) error {// 实现     return nil}  // 工厂函数:输入类型名,输出 Payment 接口 func NewPayment(kind string) Payment {switch kind {     case "alipay":         return &Alipay{}     case "wechat":         return &WechatPay{}     default:         return nil // 或 panic,视错误策略而定} }
  • 调用方只需 NewPayment("alipay"),无需 import 具体实现包
  • 新增支付方式时,只改工厂函数内部 switch,不碰已有调用代码
  • 注意:返回 nil 可能引发 panic,建议配合 error 返回或使用指针 +ok 模式

用结构体工厂支持配置化与依赖注入

当创建对象需要传入配置(如 API key、超时时间)或依赖其他服务(如日志器、数据库连接),函数型工厂不够灵活。此时定义一个带字段的工厂结构体更清晰。

type PaymentFactory struct {Logger *zap.Logger     Timeout time.Duration}  func (f *PaymentFactory) New(kind string, config map[string]string) (Payment, error) {switch kind {     case "alipay":         return &Alipay{             Key:     config["key"],             Logger:  f.Logger,             Timeout: f.Timeout,         }, nil     case "wechat":         return &WechatPay{AppID:   config["appid"],             Logger:  f.Logger,             Timeout: f.Timeout,         }, nil     default:         return nil, fmt.Errorf("unknown payment kind: %s", kind)     } }
  • 工厂自身可携带共享依赖(LoggerTimeout),避免每个创建都重复传参
  • config map[string]string 提供扩展性,不同子类型按需解析自己关心的字段
  • 返回 error 而非 nil,让调用方必须显式处理创建失败场景

工厂方法和接口组合容易被忽略的坑

Go 的接口是隐式实现,但工厂返回接口时,若忘记加 *(即返回值类型是 Alipay 而非 *Alipay),会导致方法集不匹配——因为只有指针才拥有接收者为指针的方法。

立即学习go 语言免费学习笔记(深入)”;

  • 检查你的 struct 方法接收者:如果 Processfunc (p *Alipay) Process(……),那工厂必须返回 &Alipay{},不能是 Alipay{}
  • 工厂函数签名别写成 func() *Payment —— 接口不能取地址,*Payment 是非法类型
  • 测试工厂时,别只测能否创建,要验证返回对象是否真能调用接口方法(尤其涉及嵌入字段或组合时)

工厂不是银弹。当类型分支极少(仅 2–3 种)、生命周期极短、或创建逻辑本身无状态时,直接 new 更直白。真正需要工厂的,是那些创建成本高、依赖复杂、或未来大概率要横向扩展的组件。

星耀云
版权声明:本站原创文章,由 星耀云 2026-01-01发表,共计1817字。
转载说明:转载本网站任何内容,请按照转载方式正确书写本站原文地址。本站提供的一切软件、教程和内容信息仅限用于学习和研究目的;不得将上述内容用于商业或者非法用途,否则,一切后果请用户自负。本站信息来自网络,版权争议与本站无关。
text=ZqhQzanResources