命令模式通过封装请求为对象实现解耦,支持参数化、队列、日志与撤销操作。Go中以接口定义命令,结构体实现具体逻辑,接收者执行动作,调用者触发命令,客户端组装对象。示例中灯的开关操作被封装为命令,遥控器作为调用者执行命令。扩展Undo方法可支持撤销,宏命令组合多个命令实现批量操作,闭包方式则简化函数式命令定义。该模式适用于任务队列、操作回放、UI事件处理等场景,Go的接口与组合机制使其实现更灵活简洁。

在 Golang 中实现命令模式(Command Pattern),核心是将“请求”封装成独立对象,使得可以用不同的请求对客户进行参数化,支持请求的排队、记录日志,甚至撤销操作。这种设计模式属于行为型模式,强调请求发送者与接收者之间的解耦。
命令模式的基本结构
命令模式包含以下几个关键角色:
- Command(命令接口):定义执行操作的接口,通常为一个 Execute 方法。
- ConcreteCommand(具体命令):实现命令接口,持有接收者对象,并在 Execute 中调用接收者的方法。
- Receiver(接收者):真正执行请求的对象,包含业务逻辑。
- Invoker(调用者):持有命令对象,通过调用命令的 Execute 方法来执行请求。
- Client(客户端):创建命令对象并绑定接收者,同时将命令设置给调用者。
下面是一个简单的 Go 实现示例:
立即学习“go语言免费学习笔记(深入)”;
package mainimport "fmt"
// Receiver:真正执行操作的对象type Light struct {}
func (l *Light) TurnOn() { fmt.Println("Light is on") }
func (l *Light) TurnOff() { fmt.Println("Light is off") }
// Command 接口type Command interface { Execute() }
// 具体命令:开灯type LightOnCommand struct { light *Light }
func (c *LightOnCommand) Execute() { c.light.TurnOn() }
// 具体命令:关灯type LightOffCommand struct { light *Light }
func (c *LightOffCommand) Execute() { c.light.TurnOff() }
// Invoker:遥控器type RemoteControl struct { command Command }
func (r *RemoteControl) PressButton() { r.command.Execute() }
func main() { light := &Light{} onCmd := &LightOnCommand{light: light} offCmd := &LightOffCommand{light: light}
remote := &RemoteControl{} remote.command = onCmd remote.PressButton() // 输出:Light is on remote.command = offCmd remote.PressButton() // 输出:Light is off}
支持撤销操作的命令模式
命令模式的优势之一是能轻松支持撤销(Undo)。只需在 Command 接口中增加 Undo 方法即可。
立即学习“go语言免费学习笔记(深入)”;
type Command interface { Execute() Undo() }func (c *LightOnCommand) Undo() { c.light.TurnOff() }
func (c *LightOffCommand) Undo() { c.light.TurnOn() }
调用者可以记录最近执行的命令,在需要时调用 Undo 方法回退操作。
实际应用场景
命令模式在以下场景中非常实用:
- 任务队列:将命令对象放入队列,由后台协程逐个处理,适合异步任务系统。
- 操作日志与恢复:序列化命令对象,可用于故障恢复或重放操作。
- 宏命令(Macro Command):组合多个命令为一个整体执行,例如“一键回家模式”关闭所有电器。
- UI 操作封装:菜单项、按钮点击等统一通过命令触发,便于扩展和测试。
例如,实现一个宏命令:
立即学习“go语言免费学习笔记(深入)”;
type MacroCommand struct { commands []Command }func (m *MacroCommand) Execute() { for _, cmd := range m.commands { cmd.Execute() } }
func (m *MacroCommand) Undo() { for i := len(m.commands) - 1; i >= 0; i-- { m.commands[i].Undo() } }
Go 语言中的灵活性优势
Go 虽然没有类继承,但通过接口和组合能更简洁地实现命令模式。函数式编程特性也允许直接将函数作为命令封装。
例如,使用闭包简化命令定义:
立即学习“go语言免费学习笔记(深入)”;
type FuncCommand struct { executeFunc func() undoFunc func() }func (f FuncCommand) Execute() { f.executeFunc() } func (f FuncCommand) Undo() { f.undoFunc() }
// 创建命令 onCmd := &FuncCommand{ executeFunc: light.TurnOn, undoFunc: light.TurnOff, }
这种方式更轻量,适合简单场景。
基本上就这些。Golang 的接口隐式实现和结构体组合让命令模式既规范又灵活,适用于需要解耦操作请求与执行的系统设计。










