命令模式

2020-06-22  本文已影响0人  nieniemin

什么是命令模式

Command命令模式:将一个请求封装为一个对象,从而使你可用不同的请求对客户进行参数化;对请求排队或记录请求日志,以及支持可撤销的操作。

模式角色

1.抽象命令类(Command)角色:声明执行命令的接口,拥有执行命令的抽象方法 execute()。
2.具体命令角色(DaNiaoCommand)角色:是抽象命令类的具体实现类,它拥有接收者对象,并通过调用接收者的功能来完成命令要执行的操作。持有对调用者的应用。
3.实现者/接收者(BarBecuerReceiver)角色:执行命令功能的相关操作,是具体命令对象业务的真正实现者。持有对命令的引用。
4.调用者/请求者(Invoker)角色:是请求的发送者,它通常拥有很多的命令对象,并通过访问命令对象来执行相关请求,它不直接访问接收者。

UML类图

命令模式

代码实现

以大话设计模式书中吃烧烤为例,命令Command有一个执行方法(execute)和一个撤销方法(undo);具体命令角色DaNiaoCommand实现了命令角色,可以点烧烤,吃饱了后未烤的串还可以撤销掉。下单、取消菜单我们并不会直接去后厨找烧烤师傅(接受者角色)说明,一是容易被打,二是有很多桌客户,烧烤师傅根本不知道你点的是什么。不是强耦合关系。是通过服务员(请求者角色)下单,取消菜单来通知烧烤师傅烧烤。一个服务员肯定不能只服务于一桌客户(只接受一个命令),店家会赔死的。所以服务员会服务于多桌客户(多个Command集合)。实现代码如下:

public interface Command {

    // 执行动作
    void execute();
    // 撤销操作
    void undo();
}
public class DaNiaoCommand implements Command {

    // 持有一个对接收者的引用(这里就是烧烤师傅)
    private BarBecuerReceiver barBecuerReceiver;

    public DaNiaoCommand(BarBecuerReceiver barBecuerReceiver) {
        this.barBecuerReceiver = barBecuerReceiver;
    }

    @Override
    public void execute() {
        barBecuerReceiver.barbecue();
    }

    @Override
    public void undo() {
        barBecuerReceiver.cancelBarbecue();
    }
}

// 还会有很多其他的具体命令实现Command.......
public class BarBecuerReceiver {

    public void barbecue() {
        System.out.println("实现者角色---烧烤师傅开始烧烤");
    }

    public void cancelBarbecue() {
        System.out.println("实现者角色---客人吃饱了,菜单上剩下的串不再烤了");
    }

}
public class WaiterRevoker {

    // 通过有很多命令对象(夏天点烧烤的客户有好多桌)
    private List<Command> commandList;

    public WaiterRevoker() {
        commandList = new ArrayList<>();
    }

    /**
     * 记录客户下单的需求
     *
     * @param command
     */
    public void writeMenu(Command command){
        commandList.add(command);
    }

    /**
     * 将菜单交给厨师
     */
    public void beginBarbecue() {
        if (commandList != null && commandList.size() > 0){
            commandList.forEach(command -> command.execute());
        }
    }

    /**
     * 客户不吃了,给厨师说取消菜单上的烧烤
     */
    public void cancelBarbecue() {
        if (commandList != null && commandList.size() > 0){
            commandList.forEach(command -> command.undo());
        }
    }

}
public class Client {

    public static void main(String[] args) {
        BarBecuerReceiver barBecuerReceiver = new BarBecuerReceiver();
        Command daNiaoCommand = new DaNiaoCommand(barBecuerReceiver);

        WaiterRevoker waiterRevoker = new WaiterRevoker();
        waiterRevoker.writeMenu(daNiaoCommand);
        waiterRevoker.beginBarbecue();
        waiterRevoker.cancelBarbecue();

    }
}
实现者角色---烧烤师傅开始烧烤
实现者角色---客人吃饱了,菜单上剩下的串不再烤了

优点

缺点

上一篇下一篇

猜你喜欢

热点阅读