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

命令模式是一种行为设计模式,它将请求封装为对象,从而使你可以用不同的请求、队列或日志来参数化其他对象。在 Golang 中,虽然没有类和继承的语法,但通过接口和结构体组合,依然可以优雅地实现命令模式。下面介绍如何在 Go 中封装并实践命令模式。
命令模式的核心思想
命令模式的关键在于把“执行某操作”这个行为包装成一个独立的对象(即命令),该对象包含执行所需的所有信息:方法名、参数、接收者等。这样调用者无需知道具体执行逻辑,只需触发命令即可。
主要角色包括:
- Command 接口:定义执行操作的方法(如 Execute)
- 具体命令:实现 Command 接口,持有接收者并调用其方法
- 接收者(Receiver):真正执行业务逻辑的对象
- 调用者(Invoker):不直接调用接收者,而是通过命令对象触发操作
- 客户端:创建命令对象,并绑定接收者与调用者
使用 Go 实现命令模式的基本结构
我们以一个简单的家电控制系统为例,模拟打开/关闭电灯的操作。
立即学习“go语言免费学习笔记(深入)”;
<span style="color: #008000">// 命令接口</span>
<span style="color: #0000ff">type</span> Command <span style="color: #0000ff">interface</span> {
Execute()
}
<span style="color: #008000">// 接收者:灯</span>
<span style="color: #0000ff">type</span> Light <span style="color: #0000ff">struct</span> {}
<span style="color: #0000ff">func</span> (l *Light) On() {
fmt.Println(<span style="color: #a31515">"灯已打开"</span>)
}
<span style="color: #0000ff">func</span> (l *Light) Off() {
fmt.Println(<span style="color: #a31515">"灯已关闭"</span>)
}
<span style="color: #008000">// 具体命令:开灯</span>
<span style="color: #0000ff">type</span> LightOnCommand <span style="color: #0000ff">struct</span> {
light *Light
}
<span style="color: #0000ff">func</span> (c *LightOnCommand) Execute() {
c.light.On()
}
<span style="color: #008000">// 具体命令:关灯</span>
<span style="color: #0000ff">type</span> LightOffCommand <span style="color: #0000ff">struct</span> {
light *Light
}
<span style="color: #0000ff">func</span> (c *LightOffCommand) Execute() {
c.light.Off()
}
<span style="color: #008000">// 调用者:遥控器</span>
<span style="color: #0000ff">type</span> RemoteControl <span style="color: #0000ff">struct</span> {
command Command
}
<span style="color: #0000ff">func</span> (r *RemoteControl) PressButton() {
<span style="color: #0000ff">if</span> r.command != <span style="color: #0000ff">nil</span> {
r.command.Execute()
}
}
客户端使用示例:
<span style="color: #0000ff">func</span> main() {
light := &Light{}
onCmd := &LightOnCommand{light: light}
offCmd := &LightOffCommand{light: light}
remote := &RemoteControl{}
remote.command = onCmd
remote.PressButton() <span style="color: #008000">// 输出:灯已打开</span>
remote.command = offCmd
remote.PressButton() <span style="color: #008000">// 输出:灯已关闭</span>
}
扩展功能:支持撤销操作
命令模式的一个优势是容易支持撤销(Undo)。我们可以在命令接口中增加 Undo 方法。
<span style="color: #0000ff">type</span> Command <span style="color: #0000ff">interface</span> {
Execute()
Undo()
}
<span style="color: #0000ff">func</span> (c *LightOnCommand) Undo() {
c.light.Off()
}
<span style="color: #0000ff">func</span> (c *LightOffCommand) Undo() {
c.light.On()
}
调用者也可以记录历史命令以便撤销:
<span style="color: #0000ff">type</span> RemoteControl <span style="color: #0000ff">struct</span> {
command Command
history []Command
}
<span style="color: #0000ff">func</span> (r *RemoteControl) PressButton() {
<span style="color: #0000ff">if</span> r.command != <span style="color: #0000ff">nil</span> {
r.command.Execute()
r.history = append(r.history, r.command)
}
}
<span style="color: #0000ff">func</span> (r *RemoteControl) UndoLast() {
<span style="color: #0000ff">if</span> 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 没有严格的面向对象语法,但通过接口和组合,完全可以实现干净的命令模式封装。关键是把“动作”当作一等公民传递,从而提高系统的灵活性和可扩展性。










