命令模式(请求与具体实现解耦)

目录

前言

UML

plantuml

类图

实战代码

模板

Command

Invoker

Receiver

Client


前言

命令模式解耦了命令请求者(Invoker)和命令执行者(receiver),使得 Invoker 不再直接引用 receiver,而是依赖于抽象的命令接口。具体的命令类则直接引用 receiver,通过调用 receiver 的方法来执行命令。

解耦之后,具体命令的增删改不再影响 Invoker,同时,抽象出来的命令请求还能队列化,从而实现撤销和重试功能,或做排列组合合成复杂的命令。

在 Invoker 中,也体现了桥接模式的思想,将命令的抽象和具体的实现分离,使 Invoker 中命令的添加和扩展更加简单。不过具体的命令类并不直接实现处理逻辑,而是交给了 receiver 来实现,这也是命令模式和桥接模式最主要的区别。

UML

plantuml

复制代码
@startuml
'https://plantuml.com/class-diagram

interface Command {
    + execute() : void
}

class CommandA {
    - receiver : Receiver
    + CommandA(Receiver)
    + execute() : void
}

class CommandB {
    - receiver : Receiver
    + CommandB(Receiver)
    + execute() : void
}

class Receiver {
    + executeA() : void
    + executeB() : void
}

class Invoker {
    - commands : List<Command>
    + addCommand(Command) : void
    + execute(Command) : void
    + executes() : void
}

class Client {}

Command <|.. CommandA
Command <|.. CommandB

CommandA "1" --> "1" Receiver
CommandB "1" --> "1" Receiver

Invoker "1" --> "n" Command

Client ..> Invoker
Client ..> Command

@enduml

类图

实战代码

模板

Command

java 复制代码
public interface Command {
    void execute();
}

public class ConcreteCommand1 implements Command {
    private Receiver receiver;

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

    public void execute() {
        receiver.execute1();
    }
}


public class ConcreteCommand2 implements Command {
    private Receiver receiver;

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

    public void execute() {
        receiver.execute2();
    }
}

Invoker

java 复制代码
public class Invoker {
    private List<Command> commands = new ArrayList<Command>();
    public void addCommand(Command command){
        commands.add(command);
    }

    public void execute(Command command){
        command.execute();
    }

    public void executes(){
        for(Command command : commands){
            command.execute();
        }
        commands.clear();
    }
}

Receiver

java 复制代码
public class Receiver {
    public void execute1() {
        System.out.println("execute1.");
    }

    public void execute2() {
        System.out.println("execute2.");
    }
}

Client

java 复制代码
public class Test {
    public static void main(String[] args) {
        Receiver receiver = new Receiver();
        
        Invoker invoker = new Invoker();

        invoker.execute(new Command1(receiver));

        invoker.addAction(new Command1(receiver));
        invoker.addAction(new Command2(receiver));
        invoker.executes();
    }
}
相关推荐
8***f3953 分钟前
Spring容器初始化扩展点:ApplicationContextInitializer
java·后端·spring
r_oo_ki_e_9 分钟前
java22--常用类
java·开发语言
linweidong20 分钟前
C++ 中避免悬挂引用的企业策略有哪些?
java·jvm·c++
用户937611475816121 分钟前
并发编程三大特性
java·后端
阿在在23 分钟前
Spring 系列(二):加载 BeanDefinition 的几种方式
java·后端·spring
小当家.10530 分钟前
Maven与Gradle完整对比指南:构建工具深度解析
java·gradle·maven
neoooo38 分钟前
🍃Spring Boot 多模块项目中 Parent / BOM / Starter 的正确分工
java·后端·架构
黄金贼贼38 分钟前
2026最新java单元测试json校验器
java·单元测试·json
菜鸟的迷茫39 分钟前
为了防雪崩加了限流,结果入口先挂了
java·后端·架构
荒诞硬汉41 分钟前
数组常见算法
java·数据结构·算法