




策略接口应定义具体窄接口而非interface{},以保留编译期类型检查;推荐用注册表+工厂函数解耦策略选择,输入输出需统一封装校验,避免panic和全局依赖。
Go 没有传统 OOP 的抽象类,策略模式的核心是统一接口 + 多种实现。接口应基于行为契约设计,而非为了泛化而用 interface{}。用空接口会丢失编译期类型检查,导致运行时 panic。
正确做法是定义窄接口,只暴露策略必需的方法:
type Strategy interface {
Execute(data map[string]interface{}) (map[string]interface{}, error)
Name() string
}
Execute 参数用 map[string]interface{} 是常见折中,便于传入异构配置;若字段固定,建议用结构体(如 InputParams)提升可读性和安全性Init()、Close() 等生命周期方法——策略实例应是无状态或轻量初始化的,否则难以并发复用硬编码 switch 或一堆 if-else 判断策略名,会导致新增策略必须改主逻辑。推荐用注册表 + 工厂函数解耦:
var strategies = make(map[string]Strategy)
func Register(name string, s Strategy) {
strategies[name] = s
}
func GetStrategy(name string) (Strategy, error) {
s, ok := strategies[name]
if !ok {
return nil, fmt.Errorf("unknown strategy: %s", name)
}
return s, nil
}
init() 函数或启动时集中完成,比如 Register("r
isk_control_v1", &RiskControlV1{})
func NewRiskControl(threshold float64) Strategy
策略间输入输出格式不一致是常见痛点。不要让每个策略自己解析 JSON 或转换类型——应在调度层统一封装输入、标准化错误返回。
mapstructure.Decode 或 json.Unmarshal 转成结构体,失败直接返回 ErrInvalidInput,不交给策略处理Execute 方法内部不应 panic;所有异常路径必须转为 error 返回,上层按错误类型做降级(如 fallback 到默认策略)有人尝试用 reflect 扫描包内所有实现 Strategy 接口的类型并自动注册。这看似省事,实则带来三个问题:
显式注册虽多写一行,但意图清晰、可控性强。真正的复杂点从来不在“怎么注册”,而在于策略本身的边界划分——比如「是否该把风控规则和路由决策拆成两个策略」,这需要结合业务语义判断,不是语法能解决的。