命令模式通过将操作封装为对象,实现调用者与执行者解耦。在Go中,定义Command接口包含Execute和Undo方法,具体命令如LightOnCommand持有接收者Light并调用其On/Off方法,遥控器RemoteControl作为调用者执行命令并记录历史以支持撤销。客户端创建命令、绑定接收者并交由调用者触发,结合泛型、配置加载或事件总线可提升扩展性,适用于家电控制、CLI工具等场景。

命令模式是一种行为设计模式,它将请求封装为对象,从而使你可以用不同的请求、队列或日志来参数化其他对象。在 Golang 中,虽然没有类和继承的语法,但通过接口和结构体组合,依然可以优雅地实现命令模式。下面介绍如何在 Go 中封装并实践命令模式。
命令模式的核心思想
命令模式的关键在于把“执行某操作”这个行为包装成一个独立的对象(即命令),该对象包含执行所需的所有信息:方法名、参数、接收者等。这样调用者无需知道具体执行逻辑,只需触发命令即可。
主要角色包括:
- Command 接口:定义执行操作的方法(如 Execute)
- 具体命令:实现 Command 接口,持有接收者并调用其方法
- 接收者(Receiver):真正执行业务逻辑的对象
- 调用者(Invoker):不直接调用接收者,而是通过命令对象触发操作
- 客户端:创建命令对象,并绑定接收者与调用者
使用 Go 实现命令模式的基本结构
我们以一个简单的家电控制系统为例,模拟打开/关闭电灯的操作。
立即学习“go语言免费学习笔记(深入)”;
// 命令接口 type Command interface { Execute() } // 接收者:灯 type Light struct {} func (l *Light) On() { fmt.Println("灯已打开") } func (l *Light) Off() { fmt.Println("灯已关闭") } // 具体命令:开灯 type LightOnCommand struct { light *Light } func (c *LightOnCommand) Execute() { c.light.On() } // 具体命令:关灯 type LightOffCommand struct { light *Light } func (c *LightOffCommand) Execute() { c.light.Off() } // 调用者:遥控器 type RemoteControl struct { command Command } func (r *RemoteControl) PressButton() { if r.command != nil { r.command.Execute() } }
客户端使用示例:
func main() { light := &Light{} onCmd := &LightOnCommand{light: light} offCmd := &LightOffCommand{light: light} remote := &RemoteControl{} remote.command = onCmd remote.PressButton() // 输出:灯已打开 remote.command = offCmd remote.PressButton() // 输出:灯已关闭 }
扩展功能:支持撤销操作
命令模式的一个优势是容易支持撤销(Undo)。我们可以在命令接口中增加 Undo 方法。
type Command interface { Execute() Undo() } func (c *LightOnCommand) Undo() { c.light.Off() } func (c *LightOffCommand) Undo() { c.light.On() }
调用者也可以记录历史命令以便撤销:
type RemoteControl struct { command Command history []Command } func (r *RemoteControl) PressButton() { if r.command != nil { r.command.Execute() r.history = append(r.history, r.command) } } func (r *RemoteControl) UndoLast() { if len(r.history) > 0 { last := r.history[len(r.history)-1] last.Undo() r.history = r.history[:len(r.history)-1] } }
实际应用中的封装建议
在真实项目中,可以通过以下方式提升命令模式的实用性:
- 使用泛型命令接口:Go 1.18+ 支持泛型,可定义更通用的命令结构,比如带返回值的命令
- 结合配置加载:从 JSON 或 YAML 动态加载命令映射,实现插件式控制逻辑
- 集成到 CLI 工具:Golang 常用于构建命令行工具,可用命令模式组织子命令(如 git clone / git push)
- 配合事件总线:命令触发后发布事件,实现解耦的响应流程
基本上就这些。Golang 没有严格的面向对象语法,但通过接口和组合,完全可以实现干净的命令模式封装。关键是把“动作”当作一等公民传递,从而提高系统的灵活性和可扩展性。










