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

目录

前言

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();
    }
}
相关推荐
翊谦1 小时前
Java Agent开发 Milvus 向量数据库安装
java·数据库·milvus
晓晓hh1 小时前
JavaSE学习——迭代器
java·开发语言·学习
查古穆1 小时前
栈-有效的括号
java·数据结构·算法
Java面试题总结1 小时前
Spring - Bean 生命周期
java·spring·rpc
硅基诗人2 小时前
每日一道面试题 10:synchronized 与 ReentrantLock 的核心区别及生产环境如何选型?
java
014-code2 小时前
String.intern() 到底干了什么
java·开发语言·面试
摇滚侠2 小时前
JAVA 项目教程《苍穹外卖-12》,微信小程序项目,前后端分离,从开发到部署
java·开发语言·vue.js·node.js
楚国的小隐士2 小时前
为什么说Rust是对自闭症谱系人士友好的编程语言?
java·rust·编程·对比·自闭症·自闭症谱系障碍·神经多样性
春花秋月夏海冬雪3 小时前
代码随想录刷题 - 贪心Part1
java·算法·贪心·代码随想录
野生技术架构师3 小时前
2026年牛客网最新Java面试题总结
java·开发语言