Bootstrap

Java实现命令模式

一、简介

1、定义和原理

  命令模式(Command Pattern)是一种行为设计模式,将请求(命令)封装成对象,可以用不同的请求封装不同的请求接受者,从而命令不同的接受者执行相应操作;也可以用不同的请求,调用命令对象的不同方法,实现命令对象参数化。通过定义命令接口和具体命令类,将请求封装为对象,并将其传递给调用者。

2、命令模式的结构

命令模式涉及以下几个角色:
Receiver(接收者):命令接受者,处理命令(请求)。
Command(命令):命令接口。
ConcreteCommand(具体命令):将一个请求绑定于一个接受者,调用接收者的相应操作,以实现execute操作。
Invoker(调用者):封装命令执行对应请求。
Client(客户):创建具体命令对象并设定其接收者,用调用者进行调用。

3、优缺点

(1)优点
  • 解耦:发送者和接收者解耦。
  • 灵活性:可以动态地创建、撤销请求。
(2)缺点
  • 增加类的数量:具体命令或者命令接受者需要一个单独的类来实现,这可能会导致类的数量增加。
  • 复杂性:对于简单的操作,使用命令模式可能会显得过于复杂。

二、Java实现案例

1、操作系统命令实现案例

接受者:

/**
 * @Description: 接受者
 * @Date: 2025-01-08 9:29
 * @Author: gaoyufei
 **/
public class Receiver {
    public void receive(String command){
        System.out.println("操作系统接受命令:"+command);
    }
}

命令接口:

/**
 * @Description: 命令接口
 * @Date: 2025-01-08 9:30
 * @Author: gaoyufei
 **/
public interface Command {
    void execute(String command);
    void setReceiver(Receiver receiver);
}

具体命令:

/**
 * @Description: 命令接口实现 具体命令
 * @Date: 2025-01-08 9:31
 * @Author: gaoyufei
 **/
public class ConcreteCommand implements Command {
    private Receiver receiver;
    public void setReceiver(Receiver receiver){
        this.receiver=receiver;
    }
    @Override
    public void execute(String command) {
        receiver.receive(command);
    }
}

调用者:

/**
 * @Description: 调用者
 * @Date: 2025-01-08 9:33
 * @Author: gaoyufei
 **/
public class Invoke {
    private Command command;
    public void setCommand(Command command) {
        this.command = command;
    }
    public void invoke(String command){
        this.command.execute(command);
    }
}

客户端:

/**
 * @Description: 客户端
 * @Date: 2025-01-08 9:34
 * @Author: gaoyufei
 **/
public class Client {
    public static void main(String[] args) {
        Receiver receiver=new Receiver();
        Command command=new ConcreteCommand();
        command.setReceiver(receiver);
        Invoke invoke=new Invoke();
        invoke.setCommand(command);
        invoke.invoke("打开文件");
        invoke.invoke("关闭文件");
    }
}

2、领导命令员工完成任务案例

客户端:任务板
命令:任务
调用者:领导
接受者: 员工
员工和领导都要注册在任务板上,领导操作任务板,将任务和员工关联,发布任务。

/**
 * @Description: 员工
 * @Date: 2025-01-08 10:04
 * @Author: gaoyufei
 **/
public class Employee {
    private String name;
    public Employee(String name){
        this.name=name;
    }
    public void receiveTask(String task){
        System.out.println(this.name+"接受任务:"+task);
    }
}
/**
 * @Description: 命令接口
 * @Date: 2025-01-08 10:06
 * @Author: gaoyufei
 **/
public interface Command {
    void setEmployee(Employee employee);
    void execute(String task);
}

/**
 * @Description: 命令实现,任务
 * @Date: 2025-01-08 10:07
 * @Author: gaoyufei
 **/
public class Task implements Command{
    private Employee employee;
    @Override
    public void setEmployee(Employee employee) {
       this.employee=employee;
    }
    @Override
    public void execute(String task) {
        this.employee.receiveTask(task);
    }
}

/**
 * @Description: 领导
 * @Date: 2025-01-08 10:11
 * @Author: gaoyufei
 **/
public class Leader {
    private Command command;
    public void setCommand(Command command){
        this.command=command;
    }
    //发布任务
    public void issueTask(String task){
        this.command.execute(task);
    }
}

/**
 * @Description: 任务板
 * @Date: 2025-01-08 10:17
 * @Author: gaoyufei
 **/
public class TaskBoard {
    public static void main(String[] args) {
        Command task=new Task();
        Leader leader=new Leader();
        //给张三布置任务:实现产品列表分页功能
        Employee employee1=new Employee("张三");
        task.setEmployee(employee1);
        leader.setCommand(task);
        leader.issueTask("实现产品列表分页功能");
        //给张三布置任务:实现产品列表删除功能
        Employee employee2=new Employee("李四");
        task.setEmployee(employee2);
        leader.setCommand(task);
        leader.issueTask("实现产品列表删除功能");
    }
}

3、值日委员安排值日生任务案例

客户端:值日板
命令:任务
调用者:卫生委员
接受者: 值日生
卫生委员,在值日板写谁是值日生,谁做什么任务

/**
 * @Description: 值日生
 * @Date: 2025-01-09 8:39
 * @Author: gaoyufei
 **/
public class DutyStudent {
    private String name;
    public DutyStudent(String name){
        this.name=name;
    }
    public void receiveTask(String task){
        System.out.println(this.name+":"+task);
    }
}

/**
 * @Description: 命令接口
 * @Date: 2025-01-09 8:41
 * @Author: gaoyufei
 **/
public interface Command {
    void setDutyStudent(DutyStudent dutyStudent);
    void execute(String task);
}

/**
 * @Description: 命令实现,任务
 * @Date: 2025-01-09 8:41
 * @Author: gaoyufei
 **/
public class Task implements Command{
    private DutyStudent dutyStudent;
    @Override
    public void setDutyStudent(DutyStudent dutyStudent) {
        this.dutyStudent = dutyStudent;
    }
    @Override
    public void execute(String task) {
        this.dutyStudent.receiveTask(task);
    }
}

/**
 * @Description: 卫生委员
 * @Date: 2025-01-09 8:45
 * @Author: gaoyufei
 **/
public class ClassHealthRepresentative {
    private Command task;
    public void setTask(Command task){
        this.task=task;
    }
    public void invoke(String task){
        this.task.execute(task);
    }
}

/**
 * @Description: 任务板
 * @Date: 2025-01-09 8:46
 * @Author: gaoyufei
 **/
public class TaskBoard {
    public static void main(String[] args) {
        //卫生委员
        ClassHealthRepresentative classHealthRepresentative=new ClassHealthRepresentative();
        Command task=new Task();
        DutyStudent dutyStudent1 =new DutyStudent("小明");
        task.setDutyStudent(dutyStudent1);
        //小明是值日生,今日扫地
        classHealthRepresentative.setTask(task);
        classHealthRepresentative.invoke("今天扫地");

        DutyStudent dutyStudent2 =new DutyStudent("小花");
        task.setDutyStudent(dutyStudent2);
        //小花是值日生,今日打水
        classHealthRepresentative.setTask(task);
        classHealthRepresentative.invoke("今天打水");
    }
}

4、人通过遥控控制电视案例

这是一个简单的电视遥控器系统,可以实现打开、关闭和调整音量的功能。
TV 类:接受者,封装请求相关一系列的操作。
Command 接口:定义了 execute 和 undo 方法,所有具体命令都需要实现这两个方法。
TVOffCommand、TVOnCommand、TVDownVolumeCommand、TVUpVolumeCommand 类:命令具体实现,每个命令类封装了一个接收者对象,并调用接收者的相应方法。
RemoteControl 类:调用者(遥控器),它可以绑定命令对象,并在接收到请求时调用相应的命令对象的 execute 方法。
People类:客户端,通过遥控器操作TV。

/**
 * @Description: 接受者 电视
 * @Date: 2025-01-09 9:47
 * @Author: gaoyufei
 **/
public class TV {
    private boolean status = false;
    private Integer volume = 3;
    public void on() {
        status = true;
        System.out.println("电视已打开");
    }
    public void off() {
        status = false;
        System.out.println("电视已关闭");
    }
    public void valumeUp() {
        volume++;
        System.out.println("音量调到了" + volume);
    }
    public void valumeDown() {
        volume--;
        System.out.println("音量调到了" + volume);
    }
}

/**
 * @Description: 命令接口
 * @Date: 2025-01-09 9:55
 * @Author: gaoyufei
 **/
public interface Command {
    void setTV(TV tv);
    void execute();
    void undo();
}

/**
 * @Description: 关机
 * @Date: 2025-01-09 10:02
 * @Author: gaoyufei
 **/
public class TVOffCommand implements Command{
    private TV tv;
    @Override
    public void setTV(TV tv) {
        this.tv=tv;
    }
    @Override
    public void execute() {
        this.tv.off();
    }
    @Override
    public void undo() {
        this.tv.on();
    }
}

/**
 * @Description: 开机
 * @Date: 2025-01-09 9:56
 * @Author: gaoyufei
 **/
public class TVOnCommand implements Command{
    private TV tv;
    @Override
    public void setTV(TV tv) {
        this.tv=tv;
    }
    @Override
    public void execute() {
        tv.on();
    }
    @Override
    public void undo() {
        tv.off();
    }
}

/**
 * @Description: 调低音量
 * @Date: 2025-01-09 10:05
 * @Author: gaoyufei
 **/
public class TVDownVolumeCommand implements Command{
    private TV tv;
    @Override
    public void setTV(TV tv) {
        this.tv=tv;
    }
    @Override
    public void execute() {
        this.tv.valumeDown();
    }
    @Override
    public void undo() {
        this.tv.valumeUp();
    }
}

/**
 * @Description: 调高音量
 * @Date: 2025-01-09 10:05
 * @Author: gaoyufei
 **/
public class TVUpVolumeCommand implements Command{
    private TV tv;
    @Override
    public void setTV(TV tv) {
        this.tv=tv;
    }
    @Override
    public void execute() { this.tv.valumeUp(); }
    @Override
    public void undo() {
        this.tv.valumeDown();
    }
}

/**
 * @Description: 遥控
 * @Date: 2025-01-09 10:08
 * @Author: gaoyufei
 **/
public class RemoteControl {
    private Command command;
    public void setCommand(Command command){
        this.command=command;
    }
    public void invode(){
        command.execute();
    }
    public void undo(){
        command.undo();
    }
}

/**
 * @Description: 人
 * @Date: 2025-01-09 10:11
 * @Author: gaoyufei
 **/
public class People {
    public static void main(String[] args) {
        TV tv=new TV();
        RemoteControl remoteControl=new RemoteControl();
        //开启
        Command on=new TVOnCommand();
        on.setTV(tv);
        remoteControl.setCommand(on);
        remoteControl.invode();
        //调大音量
        Command upVolume=new TVUpVolumeCommand();
        upVolume.setTV(tv);
        remoteControl.setCommand(upVolume);
        remoteControl.invode();
        remoteControl.undo();
//        //调小音量
//        Command downVolume=new TVDownVolumeCommand();
//        downVolume.setTV(tv);
//        remoteControl.setCommand(downVolume);
//        remoteControl.invode();
        //关闭
        Command off=new TVOffCommand();
        off.setTV(tv);
        remoteControl.setCommand(off);
        remoteControl.invode();
        remoteControl.undo();
    }
}
;