策略接口定义行为契约(如PaymentStrategy),具体策略类型实现接口方法,通过依赖注入在上下文中切换,避免硬编码和逻辑耦合,支持动态注册与生命周期管理。
策略模式的核心是把算法的定义和使用分离,Golang 里用接口定义策略契约,用结构体实现具体行为。关键不是“多态”本身,而是让调用方不感知具体策略细节,只依赖 Do() 这类统一方法。
比如一个支付策略接口:
type PaymentStrategy interface { Pay(amount float64) error }
两个实现:
type Alipay struct{}
func (a Alipay) Pay(amount float64) error {
fmt.Printf("Alipay: %.2f\n", amount)
return nil
}
type WechatPay struct{}
func (w WechatPay) Pay(amount float64) error {
fmt.Printf("WechatPay: %.2f\n", amount)
return nil
}
Alipay{}),也可带配置(如 type StripePay struct { apiKey string })Pay,别混进 Refund 或 Query
上下文(Context)不是必须叫 Context,它只是持有策略接口并调用其方法的结构体。策略实例应在创建上下文时传入,而不是在方法内硬编码或全局单例初始化。
type OrderProcessor struct {
strategy PaymentStrategy
}
func NewOrderProcessor(s PaymentStrategy) *OrderProcessor {
return &OrderProcessor{strategy: s}
}
func (o *OrderProcessor) Process(amount float64) error {
return o.strategy.Pay(amount)
}
使用时可自由切换:
processor := NewOrderProcessor(Alipay{})
processor.Process(99.9)
processor = NewOrderProcessor(WechatPay{})
processor.Process(128.5)
NewOrderProcessor 内部用 switch 或配置判断选哪个策略——那又把逻辑耦合回去了NewOrderProcessor(NewStripePay("sk_test_..."))
Process 方法里做策略选择,那是上层业务逻辑该干的事当策略数量变多、且需根据配置项(如 YAML 中的 payment_method: alipay)自动加载时,可以用 map 做简易注册表。注意这不是 DI 容器,只是避免 if/else 链。
var strategies = map[string]PaymentStrategy{
"alipay": Alipay{},
"wechat": WechatPay{},
"stripe": StripePay{apiKey: os.Getenv("STRIPE_KEY")},
}
func GetStrategy(name string) (PaymentStrategy, error) {
s, ok := strategies[name]
if !ok {
return nil, fmt.Errorf("unknown strategy: %s", name)
}
return s, nil
}
然后结合上下文使用:
name := "alipay" // 来自 config 或 HTTP header s, _ := GetStrategy(name) p := NewOrderProcessor(s) p.Process(199.0)
strategies map 的 value 必须是具体策略值(Alipay{})或指针(&Alipay{}),不能是未初始化的 nil 接口map[string]func() PaymentStrategy
你确实可以用 type PayFunc func(float64) error,也简洁。但接口更适合策略模式,因为:
SupportCurrency() []string 或 MinAmount() float64),函数类型无法扩展func(...) 类型断言或包装,mock 框架支持弱s.Pay(...) 点进去能看到所有实现,函数变量点进去只看到签名函数类型适合一次性、无状态、单行为的场景;策略模式本质是“可插拔的有状态行为模块”,接口才是自然表达。
真正容易被忽略的是:策略的生命周期管理。比如一个 DatabaseBackupStrategy 持有 *sql.DB,它不该被多个上下文共享,也不该在每次 Process() 时新建——得看它是无状态(可复用)还是有状态(需每次 new)。这点没想清楚,后面并发或资源泄漏就来了。