规格模式通过将业务规则封装为独立对象,支持逻辑组合,提升代码可读性和可维护性。定义Specification接口,实现IsSatisfiedBy方法,针对订单等类型创建具体规则如金额、状态、VIP判断,通过And、Or、Not组合构建复合条件,适用于复杂筛选场景。使用泛型可增强类型安全,辅助函数简化组合,规则可单独测试,便于扩展和维护。

在 Go 语言中实现规格模式(Specification Pattern),可以有效解耦业务规则的组合与判断逻辑,特别适用于复杂条件筛选或校验场景。该模式通过将每个业务规则封装为独立的“规格”对象,支持通过逻辑运算(与、或、非)组合多个规则,提升代码可读性和可维护性。
规格模式核心接口设计
定义一个通用的 Specification 接口,包含判断对象是否满足条件的方法:
type Specification interface {
IsSatisfiedBy(item interface{}) bool
}
实际使用中,可针对具体类型定义泛型化或具体类型的规格。例如对订单进行规则判断:
type Order struct {
Amount float64
Status string
IsVIP bool
}
实现基础规格与组合逻辑
每个具体规则实现 Specification 接口。同时提供组合操作(如 And、Or、Not)来构建复合规则。
立即学习“go语言免费学习笔记(深入)”;
type AndSpecification struct {
left, right Specification
}
func (a *AndSpecification) IsSatisfiedBy(item interface{}) bool {
return a.left.IsSatisfiedBy(item) && a.right.IsSatisfiedBy(item)
}
type OrSpecification struct {
left, right Specification
}
func (o *OrSpecification) IsSatisfiedBy(item interface{}) bool {
return o.left.IsSatisfiedBy(item) || o.right.IsSatisfiedBy(item)
}
type NotSpecification struct {
spec Specification
}
func (n *NotSpecification) IsSatisfiedBy(item interface{}) bool {
return !n.spec.IsSatisfiedBy(item)
}
定义具体规则:
type AmountGreaterThan struct {
limit float64
}
func (a *AmountGreaterThan) IsSatisfiedBy(item interface{}) bool {
if order, ok := item.(*Order); ok {
return order.Amount > a.limit
}
return false
}
type StatusIs struct {
status string
}
func (s *StatusIs) IsSatisfiedBy(item interface{}) bool {
if order, ok := item.(*Order); ok {
return order.Status == s.status
}
return false
}
type IsVIP struct{}
func (v *IsVIP) IsSatisfiedBy(item interface{}) bool {
if order, ok := item.(*Order); ok {
return order.IsVIP
}
return false
}
组合使用业务规则
通过组合不同规格,实现灵活的业务判断。例如:筛选“VIP 用户且订单金额大于 1000,或订单状态为已锁定”的订单:
vipAndLargeOrder := &AndSpecification{
left: &IsVIP{},
right: &AmountGreaterThan{limit: 1000},
}
orCondition := &OrSpecification{
left: vipAndLargeOrder,
right: &StatusIs{status: "locked"},
}
// 判断订单是否满足条件
order := &Order{Amount: 1200, Status: "pending", IsVIP: true}
if orCondition.IsSatisfiedBy(order) {
// 满足复合规则
}
这种结构让业务规则清晰可读,新增或调整规则时无需修改原有逻辑,只需组合新的规格对象。
优化建议与扩展
为提升类型安全和易用性,可做以下改进:
- 使用泛型(Go 1.18+)替代 interface{},避免类型断言错误
- 封装辅助函数简化组合,如 And(left, right Spec) Spec
- 支持规则链(Chain of Specification)用于校验流程
- 结合策略模式,动态选择规则组合
基本上就这些。规格模式在订单、权限、风控等多条件判断场景中非常实用,Go 虽无继承,但通过接口和组合一样能优雅实现。关键是把规则拆细,再灵活拼装。不复杂但容易忽略的是:组合结构的可测试性——每个小规格单独测,组合逻辑也清晰可验。










