状态模式通过接口与结构体实现状态分离,避免冗余条件判断。定义OrderState接口及OrderContext上下文,各状态如PendingPaymentState、PaidState等实现对应行为,调用时由当前状态决定逻辑,支持清晰的状态流转与扩展,适用于订单系统等场景。

状态模式在Golang中是一种非常实用的行为设计模式,适用于对象行为随内部状态改变而变化的场景。通过将每个状态封装为独立的结构体,并实现统一接口,可以避免大量条件判断语句(如 if/else 或 switch),提升代码可读性和可维护性。
定义状态接口与上下文对象
要实现状态模式,首先要定义一个状态接口,声明所有可能的状态行为。然后创建一个上下文对象(Context),它持有一个当前状态的引用,并将状态相关操作委托给该状态实例。
例如,假设我们正在实现一个订单系统,订单有“待支付”、“已支付”、“已发货”、“已完成”等状态:
示例代码:
type OrderState interface {
Pay(*OrderContext)
Ship(*OrderContext)
Complete(*OrderContext)
}
type OrderContext struct {
State OrderState
ID string
}
每个状态方法接收上下文指针,以便在状态变更时修改上下文中的状态字段。
立即学习“go语言免费学习笔记(深入)”;
实现具体状态结构体
每个具体状态实现 OrderState 接口,并根据业务逻辑决定是否执行操作或转换状态。比如“待支付”状态下只能进行支付操作,其他操作无效或报错。
示例:待支付状态
type PendingPaymentState struct{}
func (s *PendingPaymentState) Pay(ctx *OrderContext) {
fmt.Printf("订单 %s 支付成功\n", ctx.ID)
ctx.State = &PaidState{}
}
func (s *PendingPaymentState) Ship(ctx *OrderContext) {
fmt.Println("无法发货:订单尚未支付")
}
func (s *PendingPaymentState) Complete(ctx *OrderContext) {
fmt.Println("无法完成:订单尚未发货")
}
已支付状态示例:
type PaidState struct{}
func (s *PaidState) Pay(ctx *OrderContext) {
fmt.Println("订单已支付,无需重复支付")
}
func (s *PaidState) Ship(ctx *OrderContext) {
fmt.Printf("订单 %s 已发货\n", ctx.ID)
ctx.State = &ShippedState{}
}
func (s *PaidState) Complete(ctx *OrderContext) {
fmt.Println("无法完成:尚未发货")
}
类似地,你可以继续实现 ShippedState 和 CompletedState,形成完整状态流转链。
初始化和使用状态机
创建上下文时,设置初始状态即可开始使用。调用方法会自动根据当前状态执行对应逻辑,并可能触发状态转移。
使用示例:
func main() {
order := &OrderContext{
State: &PendingPaymentState{},
ID: "12345",
}
order.State.Pay(order) // 转为已支付
order.State.Ship(order) // 转为已发货
order.State.Complete(order) // 转为已完成
}
输出结果:
订单 12345 支付成功 订单 12345 已发货 订单已完成
整个过程无需判断当前状态,行为由实际状态对象决定,逻辑清晰且易于扩展。
状态模式使用技巧与注意事项
在实际项目中使用状态模式时,注意以下几点能提高代码质量:
- 避免状态操作暴露过多细节:建议在上下文中封装状态方法,比如提供 order.Pay() 而非直接调用 order.State.Pay(order),这样更符合面向对象习惯。
- 支持状态进入/退出钩子:可在状态接口中增加 Enter() 和 Exit() 方法,在状态切换前后执行初始化或清理逻辑。
- 结合事件驱动机制:对于复杂状态机,可引入事件类型,让状态根据事件决定转移路径,进一步解耦。
- 使用工厂函数管理状态创建:当状态带配置或依赖时,用 NewXXXState() 函数统一创建,便于测试和替换。
- 防止非法状态转移:在关键操作前加入校验,或通过表驱动方式定义合法转移规则,增强健壮性。
基本上就这些。Golang虽然没有类继承,但通过接口和组合完全可以优雅实现状态模式,尤其适合订单、工作流、连接管理等需要状态控制的场景。










