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

开关不一定是开关灯用 - 命令模式(Command Pattern)

命令模式(Command Pattern)

  • 命令模式(Command Pattern)
    • 命令设计模式
      • 命令设计模式结构图
      • 命令设计模式涉及的角色
    • talk is cheap, show you my code
    • 总结

命令模式(Command Pattern)

命令模式(Command Pattern)是一种行为型设计模式,它将请求封装为对象,从而使你可以用不同的请求对客户进行参数化、队列或者日志请求,以及支持可撤销的操作。命令模式的核心思想是将“请求”作为一个对象来处理,将发出请求的责任和执行请求的责任分割开。

太抽象了,我们还是举个现实中的例子
假如你现在在装修家,你很有可能要买一个开关。但是我们这个开关不一定是用来开关灯的,也有可能是用来控制别的东西,所以这个开关不一定是跟灯是关联的。我们的开关你可以当成我们面向对象设计里面的一个对象,它不跟灯有直接的耦合关系。开关相当于发送者,开关与什么用电器关联,开关就控制哪一个用电器。实际上你可以这样理解命令模式,通过使用命令模式,我们隔离发送请求的与执行请求的对象。

命令设计模式

命令设计模式结构图

在这里插入图片描述

命令设计模式涉及的角色

  1. 命令接口(Command):声明了所有具体命令必须实现的方法
public interface Command {
    void execute();
    void undo(); // 可选,用于撤销操作
}
  1. 具体命令(ConcreteCommand):实现了Command接口的具体类。每个具体命令都包含对一个接收者的引用,并负责调用接收者的方法来执行实际的操作。
public class ConcreteCommand implements Command {
    private final Receiver receiver;

    public ConcreteCommand(Receiver receiver) {
        this.receiver = receiver;
    }

    @Override
    public void execute() {
        receiver.performAction();
    }

    @Override
    public void undo() {
        receiver.undoAction(); // 假设接收者有撤销功能
    }
}
  1. 接收者(Receiver):接收者知道如何执行与请求相关的操作。任何类都可以成为接收者,只要它提供了可以被命令对象调用的方法。
public class Receiver {
    public void performAction() {
        System.out.println("Executing action in Receiver");
    }

    public void undoAction() {
        System.out.println("Undoing action in Receiver");
    }
}
  1. 请求发送者(Invoker):请求发送者持有一个命令对象,并负责调用命令对象的execute()方法来触发请求。发送者不需要知道具体的接收者是谁,也不需要知道请求的具体内容。
public class Invoker {
    private Command command;

    public void setCommand(Command command) {
        this.command = command;
    }

    public void executeCommand() {
        if (command != null) {
            command.execute();
        }
    }

    public void undoCommand() {
        if (command != null) {
            command.undo();
        }
    }
}
  1. 客户端(Client):客户端创建具体的命令对象,并将其与接收者关联起来。然后,客户端将命令对象传递给请求发送者,由发送者在适当的时候调用命令对象的execute()方法。
public class Client {
    public static void main(String[] args) {
        Receiver receiver = new Receiver();
        Command command = new ConcreteCommand(receiver);
        Invoker invoker = new Invoker();

        invoker.setCommand(command);
        invoker.executeCommand(); // 执行命令

        invoker.undoCommand(); // 撤销命令
    }
}

talk is cheap, show you my code

我们还是把我们举得例子转化成代码:

  1. 定义接收者(Receiver):首先,我们定义Light类,它包含打开和关闭灯光的方法。
public class Light {
    public void on() {
        System.out.println("The light is ON.");
    }

    public void off() {
        System.out.println("The light is OFF.");
    }
}
  1. 定义命令接口(Command Interface):接下来,我们定义一个命令接口Command,所有具体命令都必须实现这个接口。
public interface Command {
    void execute();
}
  1. 实现具体命令(Concrete Commands)
public class LightOnCommand implements Command {
    private final Light light;

    public LightOnCommand(Light light) {
        this.light = light;
    }

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

public class LightOffCommand implements Command {
    private final Light light;

    public LightOffCommand(Light light) {
        this.light = light;
    }

    @Override
    public void execute() {
        light.off();
    }
}
  1. 定义请求发送者(Invoker)
    现在,我们创建一个Switch类作为请求发送者。它持有一个命令对象,并负责调用命令对象的execute()方法。
public class Switch {
    private Command command;

    public void setCommand(Command command) {
        this.command = command;
    }

    public void press() {
        if (command != null) {
            command.execute();
        } else {
            System.out.println("No command set.");
        }
    }
}
  1. 客户端代码(Client)
public class Client {
    public static void main(String[] args) {
        // 创建接收者
        Light light = new Light();

        // 创建命令对象
        Command lightOn = new LightOnCommand(light);
        Command lightOff = new LightOffCommand(light);

        // 创建请求发送者并设置命令
        Switch switchButton = new Switch();
        switchButton.setCommand(lightOn);
        switchButton.press(); // 执行打开灯光命令

        switchButton.setCommand(lightOff);
        switchButton.press(); // 执行关闭灯光命令
    }
}

最后,在客户端代码中,我们将创建一个Light实例,以及相应的LightOnCommand和LightOffCommand实例。然后,我们将这些命令设置到Switch对象中,并通过调用press()方法来触发命令的执行。

总结

命令模式的主要优点包括:

  1. 降低了系统的耦合度。
  2. 新的命令可以很容易地加入到系统中。
  3. 可以方便地实现对请求的撤销和重做。
    不过,命令模式也可能增加系统的复杂性,因为它需要引入许多新的类。如果系统比较简单,使用命令模式可能会显得过于繁琐。

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

相关文章:

  • 【Rust】控制流
  • 理解AJAX与Axios:异步编程的世界
  • Java语言的正则表达式
  • 自然语言转 SQL:通过 One API 将 llama3 模型部署在 Bytebase SQL 编辑器
  • 从CentOS到龙蜥:企业级Linux迁移实践记录(龙蜥开局)
  • spring cloud的核心模块有哪些
  • HarMonyOS使用Tab构建页签
  • Megatron:深度学习中的高性能模型架构
  • LeetCode 977 题:有序数组的平方
  • Python AI教程之十八:监督学习之决策树(9) 决策树模型中的过度拟合
  • 提升租赁效率的租赁小程序全解析
  • ElasticSearch在Windows环境搭建测试
  • springcloudalibaba集成fegin报错ClassNotFoundException解决方案
  • 探索 C++ 与 LibUSB:开启 USB 设备交互的奇幻之旅
  • 47_Lua文件IO操作
  • 【计算机网络】窥探计网全貌:说说计算机网络体系结构?
  • AI语音机器人大模型是什么?
  • 如何高效格式化输出 JSON 字符串
  • 浅谈对进程的认识
  • Vue前端设置Cookie和鉴权问题
  • 为什么在二维卷积操作中,将宽度(W)维度放在高度(H)之前会破坏空间局部性原则,并影响缓存性能
  • 点赞系统设计(微服务)
  • HarmonyOS中实现TabBar(相当于Android中的TabLayout+ViewPager)
  • USA-Entrepreneur-20240708-Business/Unusual
  • Kotlin 循环语句详解
  • 数字证书管理服务