当前位置: 首页 > article >正文

【HF设计模式】06-命令模式

声明:仅为个人学习总结,还请批判性查看,如有不同观点,欢迎交流。

摘要

《Head First设计模式》第6章笔记:结合示例应用和代码,介绍命令模式,包括遇到的问题、采用的解决方案、遵循的 OO 原则、以及达到的效果。

目录

  • 摘要
  • 1 示例应用
  • 2 遇到问题
  • 3 引入设计模式
    • 3.1 改进设计
      • 3.1.1 封装变化
      • 3.1.2 针对接口编程
      • 3.1.3 新版 API 设计
      • 3.1.4 API 应用示例
      • 3.1.5 系统类图
    • 3.2 实现撤销
    • 3.3 组合命令
    • 3.4 命令模式定义
  • 4 示例代码
    • 4.1 Java 示例
    • 4.2 C++11 示例
  • 5 设计工具箱
    • 5.1 OO 基础
    • 5.2 OO 原则
    • 5.3 OO 模式
  • 参考


1 示例应用

本章的示例是,为家居自动化遥控器设计编程接口(API)。

遥控器有着众多款式。

各种遥控器

它们的共同点是:

  1. 表面分布着各种按钮,用于控制各种家居自动化设备;
  2. 包含一个全局撤销按钮,用于取消最近一次按钮操作的效果。

家居自动化设备包括电灯、风扇、音响、洒水器等等,它们由多家厂商提供,每种设备都有对应的 Java 类,用于进行自动化控制。
现有设备类的接口如下:(接口各异,并没有统一的定义)

Light
on()
off()
CeilingLight
on()
off()
dim()
CeilingFan
high()
low()
off()
getSpeed()
TV
on()
off()
setChannel()
setVolume()
GardenLight
setDuskTime()
setDawnTime()
manualOn()
manualOff()
Stereo
on()
off()
setCd()
setDvd()
setRadio()
setVolume()
Hottub
jetsOn()
jetsOff()
circulate()
setTemperature()
GarageDoor
up()
down()
stop()
lightOn()
lightOff()
Sprinkler
waterOn()
waterOff()
SecurityControl
arm()
disarm()

现在需要为遥控器创建一套 API,

  • 对于遥控器的开发者,在开发过程中,能够使用 API 配置遥控器的控制功能;
  • 对于遥控器系统,在运行过程中,能够按照开发者的配置执行控制操作。

具体要求如下:

  1. 将每个按钮与一个或一组家居设备绑定,通过按钮控制相应设备实现特定的功能。如下表所示:

    按钮编号绑定设备控制功能功能类型
    1号客厅灯开灯① 单一设备、单一操作
    2号客厅灯关灯① 单一设备、单一操作
    5号所有室内灯关灯② 多个设备、组合操作
    6号风扇高风速① 单一设备、单一操作
    9号电视打开,并设置1频道③ 单一设备、多项操作
  2. 通过撤销按钮,可以取消最近一次按钮操作(不包括撤销操作)的效果,恢复到操作前的状态;

  3. 能够支持现有的全部设备,以及厂商未来可能提供的任何设备。

遥控器 RemoteControl 类的框架如下(只包含3个方法签名),其中的 TODO 是 API 设计需要完成的内容。

public class RemoteControl {

    // TODO: 定义 API 接口,将“遥控器上的按钮”与“自动化设备的控制功能”绑定
    //       补充必要的变量和方法

    /**
     * 构造方法,在系统启动时,由硬件层触发调用
     * @param buttonCount 遥控器按钮数量(不包括撤销按钮)
     */
    RemoteControl(int buttonCount) {
        // TODO: 实现必要的初始化操作
    }

    /**
     * 按钮“按下事件”的处理方法,当按钮被按下时,由硬件层触发调用
     * 用于控制与按钮绑定的自动化设备,实现特定的功能
     * @param buttonIndex 被按下按钮的索引,从 0 开始
     */
    void buttonWasPushed(int buttonIndex) {
        // TODO: 实现控制功能
    }

    /**
     * 撤销按钮“按下事件”的处理方法,当撤销按钮被按下时,由硬件层触发调用
     * 用于撤销最近一次按钮操作(不包括撤销操作)的效果
     */
    void undoButtonWasPushed() {
        // TODO: 实现撤销功能
    }
}

2 遇到问题

在明确需求后,首先,针对最基础的“单一设备”控制功能,我们有了第1版的 API 设计:

public class RemoteControl {

    // 保存每个按钮对应的设备和操作
    private Object[] devices;
    private Integer[] operations;

    RemoteControl(int buttonCount) {
        devices = new Object[buttonCount];
        operations = new Integer[buttonCount];
    }

    // API 接口,设置按钮对应的设备和操作
    public void setButtonOperation(int buttonIndex, Object device, int operation) {
        devices[buttonIndex] = device;
        operations[buttonIndex] = operation;
    }

    // 根据按钮索引找到对应的设备,并对其进行控制
    void buttonWasPushed(int buttonIndex) {
        
        Object device = devices[buttonIndex];
        int operation = operations[buttonIndex];

        if (device instanceof Light) {
            Light light = (Light) device;              // 电灯
                 if (operation == 1) { light.on(); }   // 开灯
            else if (operation == 0) { light.off(); }  // 关灯
        } 
        else if (device instanceof CeilingFan) {
            CeilingFan ceilingFan = (CeilingFan) device;                   // 风扇
                 if (operation == CeilingFan.HIGH) { ceilingFan.high(); }  // 高速
            else if (operation == CeilingFan.LOW) { ceilingFan.low(); }    // 低速
            else if (operation == CeilingFan.OFF) { ceilingFan.off(); }    // 关闭
        }
        // 其它设备和操作(略)
    }

    void undoButtonWasPushed() {
        // TODO: 实现撤销功能
    }
}

思考题:

当前的 API 设计,违反了下面哪些设计原则?(多选)【参考答案在第 20 行】(下文 “5.2.2 原则回顾” 部分有详细些的原则介绍)

A. 分离变与不变(Identify the aspects of your application that vary and separate them from what stays the same.)
B. 针对接口编程(Program to interfaces, not implementations.)
C. 优先使用组合(Favor composition over inheritance.)
D. 松耦合设计(Strive for loosely coupled designs between objects that interact.)
E. 开闭原则(Classes should be open for extension, but closed for modification.)
F. 依赖倒置(Depend on abstractions. Do not depend on concrete classes.)











参考答案:A B D E F
参考解析:

A. 分离变与不变
   问题:在 buttonWasPushed() 方法中,设备的类型、执行的操作是变化的方面;
   改进:将变化的方面提取出来,进行封装。

B. 针对接口编程
   问题:针对 Light、CeilingFan 等具体类编程,而不是针对它们共同的接口;
   改进:定义抽象接口,针对接口编程。

D. 松耦合设计
   问题:RemoteControl 类与所有设备类强耦合;
   改进:通过接口降低耦合度。

E. 开闭原则
   问题:RemoteControl 类没有对修改关闭,每当增加新的设备类型时,都需要修改 buttonWasPushed() 方法;
   改进:访问所有设备类型共同的接口,这样在扩展新设备类型时,就不需要修改 RemoteControl 类。

F. 依赖倒置
   问题:RemoteControl 类(高层组件)依赖具体设备类(低层组件),而不是“抽象”;
   改进:定义抽象接口,使 RemoteControl 类依赖于抽象。

综上,当前设计的改进方向为:封装变化(分离变与不变)、针对接口编程。

3 引入设计模式

3.1 改进设计

参照 OO 设计原则,我们来尝试改进现有 API 的设计。

3.1.1 封装变化

buttonWasPushed() 方法中,变化的方面包括:

  • 设备对象的类型,例如 LightCeilingFanTV 等等;
  • 设备对象的操作,例如 on()high()setChannel() 等等;

需要将这些变化的方面提取出来,进行封装。

例如,将操作 on() 和执行操作的设备对象 light 提取出来,封装在一起。
由于封装后的功能是向 light 发送 on() 请求(或命令),所以将封装后的类命名为 LightOnCommand

public class LightOnCommand {
   
   // 将 light 对象封装在命令类内部
   private Light light;

   // 通过构造方法设置 light 对象,作为命令的执行者/接收者(Receiver)
   public LightOnCommand(Light light) {
      this.light = light;
   }

   // 将 on() 封装在命令方法中,每当执行命令时,就请求 light 对象执行 on() 操作
   public void execute() {
      light.on();
   }
}

类似的,将 high()ceilingFan 封装在一起,可以得到 CeilingFanHighCommand 类:

public class CeilingFanHighCommand {
   private CeilingFan ceilingFan;

   public CeilingFanHighCommand(CeilingFan ceilingFan) {
      this.ceilingFan = ceilingFan;
   }

   public void execute() {
      ceilingFan.high();
   }
}

对于需要执行“多项操作”的命令,同样可以进行封装,例如 TVOnCommand 类:

public class TVOnCommand {
   private TV tv;

   public TVOnCommand(TV tv) {
      this.tv= tv;
   }

   // 在 execute() 方法中,执行了一组操作
   public void execute() {
      tv.on();
      tv.setChannel(1);
      tv.setVolume(11);
   }
}

3.1.2 针对接口编程

接下来,遵循“针对接口编程”原则,为所有命令定义统一的 Command 接口:

public interface Command {
   // 声明 execute() 方法,用于执行命令
   public void execute();
}

每个具体命令都需要实现 Command 接口:

public class LightOnCommand implements Command { /* ... */ }
public class CeilingFanHighCommand implements Command { /* ... */ }
public class TVOnCommand implements Command { /* ... */ }

3.1.3 新版 API 设计

遵循“封装变化”、“针对接口编程”原则,改进后的 API 设计如下:

public class RemoteControl {

    // 保存每个按钮对应的命令
    private Command[] commands;

    RemoteControl(int buttonCount) {
        commands = new Command[buttonCount];
    }

    // API 接口,设置按钮对应的命令
    public void setCommand(int buttonIndex, Command command) {
        commands[buttonIndex] = command;
    }

    // 直接执行命令
    void buttonWasPushed(int buttonIndex) {
        commands[buttonIndex].execute();
    }

    void undoButtonWasPushed() {
        // TODO: 实现撤销功能
    }
}

现在,RemoteControl 作为命令的调用者(Invoker),

  1. 只需要按照 Command 接口的定义,统一调用 execute() 方法执行命令;
  2. 不再需要了解具体的设备类型、设备操作;
  3. 无论是增加新的设备类型,还是支持新的设备操作,都不需要修改 RemoteControl 类。

3.1.4 API 应用示例

遥控器 API 的应用示例如下:

public class RemoteLoader {

    public static void main(String args[]) {
        RemoteControl remoteControl = new RemoteControl(10);

        // 创建命令对象
        Light livingRoomLight = new Light("Living Room");
        LightOnCommand livingRoomLightOn = new LightOnCommand(livingRoomLight);
        LightOffCommand livingRoomLightOff = new LightOffCommand(livingRoomLight);
        
        // 将遥控器按钮与命令对象绑定
        remoteControl.setCommand(0, livingRoomLightOn);
        remoteControl.setCommand(1, livingRoomLightOff);

        // 在按钮被按下时,执行相应的命令
        remoteControl.buttonWasPushed(0);
        remoteControl.buttonWasPushed(1);
    }
}

3.1.5 系统类图

«interface»
Command
+execute()
LightOnCommand
-Light light
+execute() : CallOnOfLight
LightOffCommand
-Light light
+execute() : CallOffOfLight
Light
+on()
+off()
RemoteLoader
RemoteControl
-Command[] commands
+setCommand()
-buttonWasPushed() : CallExecuteOfCommand
  1. LightOnCommandLightOffCommand
    • 实现 Command 接口;
    • 通过封装 Light 对象及其操作,来执行请求。
      注:由于当前 mermaid 类图不支持 note,所以方法(method)的返回类型都被用于作为注释,如 CallOnOfLight
  2. RemoteControl(命令调用者)
    • 通过 Command 对象提交请求;
    • Light(命令接收者)解耦。
  3. RemoteLoader
    • 创建命令对象,并将其设置给命令调用者。

3.2 实现撤销

撤销的核心思想是:每个命令都需要能够“撤销”自己的执行效果。
撤销的实现方式为:

  1. Command 接口中声明抽象方法 undo()
  2. 由具体命令实现 undo() 方法;
    • 对于无状态撤销: 直接执行反向操作(如开灯->关灯)
    • 对于有状态撤销: 恢复之前保存的状态(如风扇速度)
  3. 保存最近一次执行的命令,在需要撤销时,调用命令的 undo() 方法。

3.2.1 声明 undo() 方法

«interface»
Command
+execute()
+undo()

3.2.2 实现 undo() 方法(无状态)

LightOnCommand 为例,执行 execute() 开灯后,如果撤销命令,对应的就是执行关灯操作。

public class LightOnCommand implements Command {
    private Light light;
    public LightOnCommand(Light light) { this.light = light; }

    public void execute() { light.on(); }

    // 直接执行反向操作
    public void undo() { light.off(); }
}

3.2.3 实现 undo() 方法(有状态)

CeilingFanOffCommand 为例,由于风扇具有多种状态,所以在执行 execute() 关闭风扇之前,需要保存风扇当前的状态,以便在撤销命令时,可以恢复到之前保存的状态。

public class CeilingFanOffCommand implements Command {
   CeilingFan ceilingFan;
   int prevSpeed;

   public CeilingFanOffCommand(CeilingFan ceilingFan) {
      this.ceilingFan = ceilingFan;
   }

   public void execute() {
      // 保存命令执行前的状态
      prevSpeed = ceilingFan.getSpeed();
      ceilingFan.off();
   }

   // 恢复到命令执行前的状态
   public void undo() {
           if (prevSpeed == CeilingFan.HIGH) { ceilingFan.high(); } 
      else if (prevSpeed == CeilingFan.LOW) { ceilingFan.low(); }
      else if (prevSpeed == CeilingFan.OFF) { ceilingFan.off(); }
   }
}

3.2.4 调用 undo() 方法

为了能够撤销最近一次执行的命令,需要先保存该命令;在执行撤销时,通过调用该命令的 undo() 方法,将系统恢复到命令执行前的状态。

public class RemoteControl {

    private Command[] commands;
    private Command undoCommand;

    void buttonWasPushed(int buttonIndex) {
        commands[buttonIndex].execute();
        undoCommand = commands[buttonIndex]; // 保存最近一次执行的命令
    }

    void undoButtonWasPushed() {
        // 调用最近一次执行命令的 undo() 方法,恢复到命令执行前的状态
        undoCommand.undo();
    }

    // 其它变量和方法(略)
}

如果需要撤销最近执行的多个命令,而不仅仅是一个命令,
可以将已经执行的命令保存在一个栈(历史表列)中,每当需要撤销命令时,就弹出栈顶命令,调用其 undo() 方法。

3.2.5 记录命令日志

与撤销功能类似,通过在命令中定义 store()load() 方法,还可以将命令写到日志中,并在需要的时候重新加载和执行命令。
(这项功能对于当前的遥控器来说,并没有意义;但是在一些数据处理应用中,通过记录日志,可以在异常发生后实现数据恢复。)

«interface»
Command
+execute()
+undo()
+store()
+load()

3.3 组合命令

为了实现“使用一个按钮,关闭所有室内灯”,我们需要引入一个新的命令类型 MacroCommand

«interface»
Command
+execute()
+undo()
MacroCommand
-Command[] commands
+execute() : CallExecuteOfEachCommand
+undo() : CallUndoOfEachCommand
  • 因为 MacroCommand 实现 Command,所以 MacroCommand 可以作为普通的具体命令使用;
  • 因为 MacroCommand 组合 Command,所以 MacroCommand 可以借助一系列的具体命令对象执行操作。

下面是 MacroCommand 类的定义:

public class MacroCommand implements Command {
    Command[] commands;

    public MacroCommand(Command[] commands) {
        this.commands = commands;
    }

    public void execute() {
        for (int i = 0; i < commands.length; i++) {
            commands[i].execute();
        }
    }

    public void undo() {
        // 按照后执行先撤销的顺序调用 undo()
        for (int i = commands.length - 1; i >= 0; i--) {
            commands[i].undo();
        }
    }
}

一键关灯的代码示意如下:

RemoteControl remoteControl = new RemoteControl(10);

Light livingRoomLight = new Light("Living Room");
Light kitchenLight = new Light("Kitchen");

LightOffCommand livingRoomLightOff = new LightOffCommand(livingRoomLight);
LightOffCommand kitchenLightOff = new LightOffCommand(kitchenLight);

Command[] lightsOffCommands = { livingRoomLightOff, kitchenLightOff };
remoteControl.setCommand(4, new MacroCommand(lightsOffCommands));

remoteControl.buttonWasPushed(4);
remoteControl.undoButtonWasPushed();

3.4 命令模式定义

刚刚,我们已经采用“命令模式”完成了家居自动化遥控器的 API 设计。下面是命令模式的正式定义:

命令模式(Command Pattern)
将一个请求封装为一个对象,从而使你可用不同的请求对客户进行参数化,对请求排队或记录请求日志,以及支持可撤销的操作。
Encapsulate a request as an object, thereby letting you parameterize clients with different requests, queue or log requests, and support undoable operations.

这里的客户(Client)指的是使用命令的用户(即调用者 Invoker),可以使用不同的命令对象来配置调用者的行为。

«interface»
Command
+execute()
+undo()
ConcreteCommand
-Receiver receiver
-state
+ConcreteCommand(Receiver)
+execute() : CallActionOfReceiver
+undo() : ReverseTheAction
Receiver
+action()
Client
Invoker
-Command command
+setCommand(Command)
+executeCommand()
+undoCommand()
  1. Command,抽象命令接口
    • 声明抽象的 execute() 方法,用于提交请求;
    • 声明抽象的 undo() 方法,用于撤销已经提交的请求(恢复到请求执行前的状态)。
  2. ConcreteCommand,具体命令实现类
    • 声明 receiver 实例变量,作为命令的接收者;
      • 如果具体命令可以独立完成请求,那么就不需要接收者;
      • 如果命令的执行还需要其它参数,那么可以声明更多的实例变量来提供参数设置。
    • 实现 execute() 方法,
      • 首先,通过 state 保存当前状态;(如果该请求是可撤销的,并且需要保存状态)
      • 然后,调用 receiver.action() 执行请求;
    • 实现 undo() 方法,恢复 state 状态,取消执行 execute() 的效果。
  3. Client
    • 创建 ConcreteCommand 对象,并指定它的 Receiver 对象;
    • Invoker 对象设置 ConcreteCommand 对象;
      • 因为 ConcreteCommand 对象可以被传递、被存储,
        所以,几经辗转之后,可能会由 OtherClient 来为 Invoker 对象设置 ConcreteCommand 对象。
    • 通过 CommandInvokerReceiver 解耦。
  4. Invoker,命令调用者
    • 声明 command 实例变量,引用 ConcreteCommand 对象;
    • 在需要提交请求时,调用 command.execute()
    • 在需要撤销请求时,调用 command.undo()。(如果该请求是可撤销的)
  5. Receiver,命令接收者
    • 定义 action() 方法,执行请求对应的具体操作。

参与者之间的交互如下:

aReceiver aClient aCommand anInvoker new Command(aReceiver) setCommand(aCommand) execute() action() aReceiver aClient aCommand anInvoker

模式效果:

  • 命令是一等对象,可以像其他对象一样被操作(如创建、传递、存储)和扩展(通过继承或组合);
  • 【优点】降低耦合度:将调用操作的对象(Invoker)与知道如何实现该操作的对象(Receiver)解耦;
  • 【优点】易于扩展:增加新的 ConcreteCommand 时,不需要改变已有的类;
  • 【缺点】增加复杂度:可能会定义过多的具体命令类,增加系统的复杂度。

延伸阅读:《设计模式:可复用面向对象软件的基础》 5.2 Command(命令)— 对象行为型模式 [P175-183]

4 示例代码

4.1 Java 示例

厂商设备类的定义:

// Light.java
public class Light {
   String location;
   public Light(String location) { this.location = location; }

   public void on() { System.out.println(location + " light is on"); }
   public void off() { System.out.println(location + " light is off"); }
}

// CeilingFan.java
public class CeilingFan {
   public static final int HIGH = 2;
   public static final int LOW = 1;
   public static final int OFF = 0;
   String location;
   int speed;

   public CeilingFan(String location) {
      this.location = location;
      speed = OFF;
   }

   public void high() {
      speed = HIGH;
      System.out.println(location + " ceiling fan is on high");
   }

   public void low() {
      speed = LOW;
      System.out.println(location + " ceiling fan is on low");
   }

   public void off() {
      speed = OFF;
      System.out.println(location + " ceiling fan is off");
   }

   public int getSpeed() {
      return speed;
   }
}

命令接口和实现类的定义:

// Command.java
public interface Command {
    public void execute();
    public void undo();
}

// NoCommand.java 空对象模式:实现命令,但不做任何操作
public class NoCommand implements Command {
    public void execute() {}
    public void undo() { }
}

// MacroCommand.java
public class MacroCommand implements Command {
    Command[] commands;

    public MacroCommand(Command[] commands) {
        this.commands = commands;
    }

    public void execute() {
        for (int i = 0; i < commands.length; i++) {
            commands[i].execute();
        }
    }

    // these commands have to be done backwards to ensure proper undo functionality
    public void undo() {
        for (int i = commands.length - 1; i >= 0; i--) {
            commands[i].undo();
        }
    }
}

// LightOnCommand.java
public class LightOnCommand implements Command {
    Light light;
    public LightOnCommand(Light light) { this.light = light; }

    public void execute() { light.on(); }
    public void undo() { light.off(); }
}

// LightOffCommand.java
public class LightOffCommand implements Command {
   Light light;
   public LightOffCommand(Light light) { this.light = light; }

   public void execute() { light.off(); }
   public void undo() { light.on(); }
}

// CeilingFanCommand.java 通过抽象基类实现 undo() 方法,在子类中实现 execute() 方法
public abstract class CeilingFanCommand implements Command {
    protected CeilingFan ceilingFan;
    protected int prevSpeed;
    public CeilingFanCommand(CeilingFan ceilingFan) { this.ceilingFan = ceilingFan; }

    @Override
    public abstract void execute();

    @Override
    public void undo() {
        switch (prevSpeed) {
            case CeilingFan.HIGH: ceilingFan.high(); break;
            case CeilingFan.LOW: ceilingFan.low(); break;
            case CeilingFan.OFF: ceilingFan.off(); break;
        }
    }
}

// CeilingFanHighCommand.java
public class CeilingFanHighCommand extends CeilingFanCommand {
    public CeilingFanHighCommand(CeilingFan ceilingFan) { super(ceilingFan); }

    public void execute() {
        prevSpeed = ceilingFan.getSpeed();
        ceilingFan.high();
    }
}

// CeilingFanLowCommand.java
public class CeilingFanLowCommand extends CeilingFanCommand {
    public CeilingFanLowCommand(CeilingFan ceilingFan) { super(ceilingFan); }

    public void execute() {
        prevSpeed = ceilingFan.getSpeed();
        ceilingFan.low();
    }
}

// CeilingFanOffCommand.java
public class CeilingFanOffCommand extends CeilingFanCommand {
    public CeilingFanOffCommand(CeilingFan ceilingFan) { super(ceilingFan); }

    public void execute() {
        prevSpeed = ceilingFan.getSpeed();
        ceilingFan.off();
    }
}

遥控器的定义:

// RemoteControl.java
public class RemoteControl {
    private Command[] commands;
    private Command undoCommand;

    RemoteControl(int buttonCount) {
        commands = new Command[buttonCount];

        // 将所有命令都初始化为一个空对象,这样在执行命令时,就不需要进行空指针检查
        Command noCommand = new NoCommand();
        for (int i = 0; i < buttonCount; i++) {
            commands[i] = noCommand;
        }
        undoCommand = noCommand;
    }

    public void setCommand(int buttonIndex, Command command) {
        commands[buttonIndex] = command;
    }

    void buttonWasPushed(int buttonIndex) {
        commands[buttonIndex].execute();
        undoCommand = commands[buttonIndex];
    }

    void undoButtonWasPushed() {
        undoCommand.undo();
    }

    public String toString() {
        StringBuffer stringBuf = new StringBuffer();
        stringBuf.append("\n------ Remote Control ------\n");
        for (int i = 0; i < commands.length; i++) {
            stringBuf.append("[button " + i + "] " + commands[i].getClass().getName() + "\n");
        }
        stringBuf.append("[undo] " + undoCommand.getClass().getName() + "\n");
        return stringBuf.toString();
    }
}

测试代码:

// RemoteLoader.java
public class RemoteLoader {

    public static void main(String args[]) {
        // 遥控器实例(Invoker)
        RemoteControl remoteControl = new RemoteControl(10);

        // 设备实例(Receiver)
        Light livingRoomLight = new Light("Living Room");
        Light kitchenLight = new Light("Kitchen");
        CeilingFan ceilingFan = new CeilingFan("Living Room");

        // 命令实例(ConcreteCommand)
        LightOnCommand livingRoomLightOn = new LightOnCommand(livingRoomLight);
        LightOffCommand livingRoomLightOff = new LightOffCommand(livingRoomLight);
        LightOnCommand kitchenLightOn = new LightOnCommand(kitchenLight);
        LightOffCommand kitchenLightOff = new LightOffCommand(kitchenLight);

        Command[] lightsOffCommands = { livingRoomLightOff, kitchenLightOff };
        MacroCommand lightsOffMacro = new MacroCommand(lightsOffCommands);

        CeilingFanHighCommand ceilingFanHigh = new CeilingFanHighCommand(ceilingFan);
        CeilingFanLowCommand ceilingFanLow = new CeilingFanLowCommand(ceilingFan);
        CeilingFanOffCommand ceilingFanOff = new CeilingFanOffCommand(ceilingFan);

        // 设置遥控器按钮对应的命令(Invoker::setCommand)
        remoteControl.setCommand(0, livingRoomLightOn);
        remoteControl.setCommand(1, livingRoomLightOff);
        remoteControl.setCommand(2, kitchenLightOn);
        remoteControl.setCommand(3, kitchenLightOff);

        remoteControl.setCommand(4, lightsOffMacro);

        remoteControl.setCommand(5, ceilingFanHigh);
        remoteControl.setCommand(6, ceilingFanLow);
        remoteControl.setCommand(7, ceilingFanOff);

        System.out.println(remoteControl);

        // 执行命令(Invoker::executeCommand)和 撤销命令(Invoker::undoCommand)
        remoteControl.buttonWasPushed(0);     // 客厅灯打开
        remoteControl.buttonWasPushed(1);     // 客厅灯关闭
        remoteControl.undoButtonWasPushed();  // 撤销关闭,客厅灯打开
        remoteControl.buttonWasPushed(2);     // 厨房灯打开
        remoteControl.buttonWasPushed(3);     // 厨房灯关闭
        remoteControl.undoButtonWasPushed();  // 撤销关闭,厨房灯打开

        remoteControl.buttonWasPushed(4);     // 关闭所有灯
        remoteControl.undoButtonWasPushed();  // 撤销关闭,打开所有灯

        remoteControl.buttonWasPushed(5);     // 风扇高速
        remoteControl.buttonWasPushed(6);     // 风扇低速
        remoteControl.undoButtonWasPushed();  // 撤销低速,风扇高速
        remoteControl.buttonWasPushed(7);     // 风扇关闭
    }
}

4.2 C++11 示例

厂商设备类的定义:

struct Light {
  Light(const std::string &location) : location(location) {}

  void on() { std::cout << location << " light is on\n"; }
  void off() { std::cout << location << " light is off\n"; }

 private:
  std::string location;
};

struct CeilingFan {
  enum class Speed { OFF, LOW, HIGH };

  CeilingFan(const std::string &location) : location(location) {}

  void high() {
    speed = Speed::HIGH;
    std::cout << location << " ceiling fan is on high\n";
  }
  void low() {
    speed = Speed::LOW;
    std::cout << location << " ceiling fan is on low\n";
  }
  void off() {
    speed = Speed::OFF;
    std::cout << location << " ceiling fan is off\n";
  }
  Speed getSpeed() const { return speed; }

 private:
  std::string location;
  Speed speed = Speed::OFF;
};

命令接口和实现类的定义:

struct Command {
  virtual ~Command() = default;
  virtual void execute() = 0;
  virtual void undo() = 0;
};

// 空对象模式:实现命令,但不做任何操作
struct NoCommand : Command {
  void execute() override {}
  void undo() override {}
};

struct MacroCommand : Command {
  MacroCommand(const std::vector<std::shared_ptr<Command>> &commands) : commands(commands) {}

  void execute() override {
    for (const auto &command : commands) {
      command->execute();
    }
  }

  // these commands have to be done backwards to ensure proper undo functionality
  void undo() override {
    for (auto it = commands.rbegin(); it != commands.rend(); ++it) {
      (*it)->undo();
    }
  }

 private:
  std::vector<std::shared_ptr<Command>> commands;
};

struct LightOnCommand : Command {
  LightOnCommand(Light &light) : light(light) {}

  void execute() override { light.on(); }
  void undo() override { light.off(); }

 private:
  Light &light;
};

struct LightOffCommand : Command {
  LightOffCommand(Light &light) : light(light) {}

  void execute() override { light.off(); }
  void undo() override { light.on(); }

 private:
  Light &light;
};

// 通过抽象基类实现 undo() 方法,在子类中实现 execute() 方法
struct CeilingFanCommand : Command {
  CeilingFanCommand(CeilingFan &ceilingFan) : ceilingFan(ceilingFan) {}

  virtual void execute() override = 0;

  void undo() override {
    switch (prevSpeed) {
      case CeilingFan::Speed::HIGH: ceilingFan.high(); break;
      case CeilingFan::Speed::LOW: ceilingFan.low(); break;
      case CeilingFan::Speed::OFF: ceilingFan.off(); break;
    }
  }

 protected:
  CeilingFan &ceilingFan;
  CeilingFan::Speed prevSpeed;
};

struct CeilingFanHighCommand : CeilingFanCommand {
  CeilingFanHighCommand(CeilingFan &ceilingFan) : CeilingFanCommand(ceilingFan) {}

  void execute() override {
    prevSpeed = ceilingFan.getSpeed();
    ceilingFan.high();
  }
};

struct CeilingFanLowCommand : CeilingFanCommand {
  CeilingFanLowCommand(CeilingFan &ceilingFan) : CeilingFanCommand(ceilingFan) {}

  void execute() override {
    prevSpeed = ceilingFan.getSpeed();
    ceilingFan.low();
  }
};

struct CeilingFanOffCommand : CeilingFanCommand {
  CeilingFanOffCommand(CeilingFan &ceilingFan) : CeilingFanCommand(ceilingFan) {}

  void execute() override {
    prevSpeed = ceilingFan.getSpeed();
    ceilingFan.off();
  }
};

遥控器的定义:

struct RemoteControl {
  explicit RemoteControl(int buttonCount) {
    // 将所有命令都初始化为一个空对象,这样在执行命令时,就不需要进行空指针检查
    auto noCommand = std::make_shared<NoCommand>();
    commands = std::vector<std::shared_ptr<Command>>(buttonCount, noCommand);
    undoCommand = noCommand;
  }

  void setCommand(int buttonIndex, std::shared_ptr<Command> command) {
    commands[buttonIndex] = command; 
  }

  void buttonWasPushed(int buttonIndex) {
    commands[buttonIndex]->execute();
    undoCommand = commands[buttonIndex];
  }

  void undoButtonWasPushed() const {
    undoCommand->undo(); 
  }

 private:
  std::vector<std::shared_ptr<Command>> commands;
  std::shared_ptr<Command> undoCommand;

  friend std::ostream &operator<<(std::ostream &os, const RemoteControl &remoteControl);
};

std::ostream &operator<<(std::ostream &os, const RemoteControl &remoteControl) {
  os << "\n------ Remote Control ------\n";
  for (int i = 0; i < remoteControl.commands.size(); ++i) {
    os << "[button " << i << "] " << typeid(*remoteControl.commands[i]).name() << "\n";
  }
  os << "[undo] " << typeid(*remoteControl.undoCommand).name() << "\n\n";
  return os;
}

测试代码:

#include <iostream>
#include <memory>
#include <string>
#include <typeinfo>
#include <vector>

// 在这里添加相关接口和类的定义

int main() {
  // 遥控器实例(Invoker)
  RemoteControl remoteControl(10);

  // 设备实例(Receiver)
  Light livingRoomLight("Living Room");
  Light kitchenLight("Kitchen");
  CeilingFan ceilingFan("Living Room");

  // 命令实例(ConcreteCommand)
  auto livingRoomLightOn = std::make_shared<LightOnCommand>(livingRoomLight);
  auto livingRoomLightOff = std::make_shared<LightOffCommand>(livingRoomLight);
  auto kitchenLightOn = std::make_shared<LightOnCommand>(kitchenLight);
  auto kitchenLightOff = std::make_shared<LightOffCommand>(kitchenLight);

  std::vector<std::shared_ptr<Command>> lightsOffCommands = {livingRoomLightOff, kitchenLightOff};
  auto lightsOffMacro = std::make_shared<MacroCommand>(lightsOffCommands);

  auto ceilingFanHigh = std::make_shared<CeilingFanHighCommand>(ceilingFan);
  auto ceilingFanLow = std::make_shared<CeilingFanLowCommand>(ceilingFan);
  auto ceilingFanOff = std::make_shared<CeilingFanOffCommand>(ceilingFan);

  // 设置遥控器按钮对应的命令(Invoker::setCommand)
  remoteControl.setCommand(0, livingRoomLightOn);
  remoteControl.setCommand(1, livingRoomLightOff);
  remoteControl.setCommand(2, kitchenLightOn);
  remoteControl.setCommand(3, kitchenLightOff);

  remoteControl.setCommand(4, lightsOffMacro);

  remoteControl.setCommand(5, ceilingFanHigh);
  remoteControl.setCommand(6, ceilingFanLow);
  remoteControl.setCommand(7, ceilingFanOff);

  std::cout << remoteControl;

  // 执行命令(Invoker::executeCommand)和 撤销命令(Invoker::undoCommand)
  remoteControl.buttonWasPushed(0);     // 客厅灯打开
  remoteControl.buttonWasPushed(1);     // 客厅灯关闭
  remoteControl.undoButtonWasPushed();  // 撤销关闭,客厅灯打开
  remoteControl.buttonWasPushed(2);     // 厨房灯打开
  remoteControl.buttonWasPushed(3);     // 厨房灯关闭
  remoteControl.undoButtonWasPushed();  // 撤销关闭,厨房灯打开

  remoteControl.buttonWasPushed(4);     // 关闭所有灯
  remoteControl.undoButtonWasPushed();  // 撤销关闭,打开所有灯

  remoteControl.buttonWasPushed(5);     // 风扇高速
  remoteControl.buttonWasPushed(6);     // 风扇低速
  remoteControl.undoButtonWasPushed();  // 撤销低速,风扇高速
  remoteControl.buttonWasPushed(7);     // 风扇关闭
}

5 设计工具箱

5.1 OO 基础

OO 基础回顾

  1. 抽象(Abstraction)
  2. 封装(Encapsulation)
  3. 继承(Inheritance)
  4. 多态(Polymorphism)

5.2 OO 原则

5.2.1 新原则

最近两章都没有介绍新的 OO 原则。

5.2.2 原则回顾

  1. 封装变化。
    Encapsulate what varies.
  2. 针对接口编程,而不是针对实现编程。
    Program to interfaces, not implementations.
  3. 优先使用组合,而不是继承。
    Favor composition over inheritance.
  4. 尽量做到交互对象之间的松耦合设计。
    Strive for loosely coupled designs between objects that interact.
  5. 类应该对扩展开放,对修改关闭。
    Classes should be open for extension, but closed for modification.
  6. 依赖抽象,不依赖具体类。
    Depend on abstractions. Do not depend on concrete classes.

5.3 OO 模式

5.3.1 新模式

命令模式(Command Pattern)

  • 把请求封装为对象,
    The Command Pattern encapsulates a request as an object,
  • 以便用不同的请求来参数化客户,对请求进行排队或记录请求日志,并支持可撤销的操作。
    thereby letting you parameterize clients with different requests, queue or log requests, and support undoable operations.

5.3.2 模式回顾

1 创建型模式(Creational Patterns)

创建型模式与对象的创建有关。
Creational patterns concern the process of object creation.

  1. 工厂方法(Factory Method)
    • 定义了一个创建对象的接口,但由子类决定要实例化哪个类。
      The Factory Method Pattern defines an interface for creating an object, but lets subclasses decide which class to instantiate.
    • 工厂方法让类把实例化推迟到子类。
      Factory Method lets a class defer instantiation to subclasses.
  2. 抽象工厂(Abstract Factory)
    • 提供一个接口,创建相关或依赖对象的家族,而不需要指定具体类。
      The Abstract Factory Pattern provides an interface for creating families of related or dependent objects without specifying their concrete classes.
  3. 单例模式(Singleton Pattern)
    • 确保一个类只有一个实例,并提供一个全局访问点。
      The Singleton Pattern ensures a class has only one instance, and provides a global point of access to it.

2 结构型模式(Structural Patterns)

结构型模式处理类或对象的组合。
Structural patterns deal with the composition of classes or objects.

  1. 装饰者模式(Decorator Pattern)
    • 动态地给一个对象添加一些额外的职责。
      The Decorator Pattern attaches additional responsibilities to an object dynamically.
    • 就增加功能来说,装饰者模式相比生成子类更为灵活。
      Decorators provide a flexible alternative to subclassing for extending functionality.

3 行为型模式(Behavioral Patterns)

行为型模式描述类或对象之间的交互方式以及职责分配方式。
Behavioral patterns characterize the ways in which classes or objects interact and distribute responsibility.

  1. 策略模式(Strategy Pattern)
    • 定义一个算法家族,把其中的算法分别封装起来,使得它们之间可以互相替换。
      Strategy defines a family of algorithms, encapsulates each one, and makes them interchangeable.
    • 让算法的变化独立于使用算法的客户。
      Strategy lets the algorithm vary independently from clients that use it.
  2. 观察者模式(Observer Pattern)
    • 定义对象之间的一对多依赖,
      The Observer Pattern defines a one-to-many dependency between objects
    • 这样一来,当一个对象改变状态时,它的所有依赖者都会被通知并自动更新。
      so that when one object changes state, all of its dependents are notified and updated automatically.

参考

  1. [美]弗里曼、罗布森著,UMLChina译.Head First设计模式.中国电力出版社.2022.2
  2. [美]伽玛等著,李英军等译.设计模式:可复用面向对象软件的基础.机械工业出版社.2019.3
  3. wickedlysmart: Head First设计模式 Java 源码

Hi, I’m the ENDing, nice to meet you here! Hope this article has been helpful.


http://www.kler.cn/a/513179.html

相关文章:

  • 微软预测 AI 2025,AI Agents 重塑工作形式
  • VS Code AI开发之Copilot配置和使用详解
  • C# 以管理员方式启动程序全解析
  • Git下载安装
  • mac m1下载maven安装并配置环境变量
  • three.js实现裸眼双目平行立体视觉
  • Flink底层架构与运行流程
  • 2.4 kubectl命令行设置7大命令分组
  • 三轴云台之跟随模式篇
  • JAVA:策略模式(Strategy Pattern)的技术指南
  • Java泛型方法所受的限制是什么?
  • JDBC实验测试
  • 软通动力携鸿湖万联与微展世签署战略合作协议,以开源鸿蒙赋能工业创新升级
  • 【深度学习基础】多层感知机 | 多层感知机的实现
  • K8S如何让worker使用kubectl命令(RBAC方法)
  • 机器学习-核函数(Kernel Function)
  • 使用xorriso v1.5.2和grub4dos 0.4.6a -2024-02-26制作可启动ISO文件
  • 《Keras 3 使用 Reptile 进行 Few-Shot 学习》
  • SSL证书的颁发格式和制作过
  • 第四天 安装DevEco Studio,配置HarmonyOS开发环境
  • 【集合】单列集合和双列集合
  • OpenCV简介、OpenCV安装
  • 25届自动化考研复试微机原理基础版题库
  • Y3编辑器2.0功能指引
  • js手写-实现Promise的实例方法
  • 深度学习中梯度的补充理解