19.设计模式-命令模式

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

需求

实现客户在烤肉店买烤肉的过程。

需求分析

  • 用例功能:1.客户下单2.通知厨师制作3.上桌吃饭4.付钱走人
  • 用例实现:只实现1,2步
  • 下单细化:两种品类,鸡翅、羊肉串
  • 动作细化:下单、撤销、修改数量

代码

业务类

c 复制代码
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

typedef enum {
    MUTTON,
    CHICHKENWING,
    COMMANDMAX
} CommandType;
// 厨师
typedef struct Barbecuer {
    void (*bake[COMMANDMAX])();
} Barbecuer;

void BakeMutton() {
    printf("烤羊肉串\n");
}

void BakeChickenWings() {
    printf("烤鸡翅\n");
}

Barbecuer *InitBarbecuer() {
    Barbecuer *obj = (Barbecuer *)malloc(sizeof(Barbecuer));
    obj->bake[MUTTON] = BakeMutton;
    obj->bake[CHICHKENWING] = BakeChickenWings;
    return obj;
}
// 命令类
typedef struct Command {
    Barbecuer *receiver;
    void (*excecute)(struct Command *);
} Command;

typedef struct BakeMuttonCommand {
    Command base;
} BakeMuttonCommand;

void ExcecuteBakeMuttonCommand(Command *obj) {
    obj->receiver->bake[MUTTON]();
}

Command *InitBakeMuttonCommand(Barbecuer *receiver) {
    BakeMuttonCommand *obj = (BakeMuttonCommand *)malloc(sizeof(BakeMuttonCommand));
    obj->base.excecute = ExcecuteBakeMuttonCommand;
    obj->base.receiver = receiver;
    return (Command *)obj;
}

typedef struct BakeChickenWingsCommand {
    Command base;
} BakeChickenWingsCommand;

void ExcecuteBakeChickenWingsCommand(Command *obj) {
    obj->receiver->bake[CHICHKENWING]();
}

Command *InitBakeChickenWingsCommand(Barbecuer *receiver) {
    BakeChickenWingsCommand *obj = (BakeChickenWingsCommand *)malloc(sizeof(BakeChickenWingsCommand));
    obj->base.excecute = ExcecuteBakeChickenWingsCommand;
    obj->base.receiver = receiver;
    return (Command *)obj;
}

typedef Command *(*CreateCommandInterface)(Barbecuer *);

CreateCommandInterface cmdinterface[COMMANDMAX] = {InitBakeMuttonCommand, InitBakeChickenWingsCommand};

typedef struct Order {
    int num;
    Barbecuer *receiver;
} Order;
// 服务员
typedef struct Waiter {
    Order order[COMMANDMAX];
    void (*setOrder)(struct Waiter *, CommandType, int, Barbecuer *);
    void (*notify)(struct Waiter *);
} Waiter;

void WaiterSetOrder(Waiter *obj, CommandType type, int num, Barbecuer *receiver) {
    obj->order[type] = (Order){num, receiver};
}

void WaiterNotify(Waiter *obj) {
    Command *cmd;
    for(int i=0; i<COMMANDMAX; i++) {
        if(obj->order[i].num != 0) {
            printf("%d串:", obj->order[i].num);
            // 1.直接烤
            obj->order[i].receiver->bake[i]();
            // 2.编成命令再烤
            // cmd = cmdinterface[i](obj->order[i].receiver);
            // cmd->excecute(cmd);
            // free(cmd);
        }
    }
}

Waiter *InitWaiter() {
    Waiter *obj = (Waiter *)malloc(sizeof(Waiter));
    obj->setOrder = WaiterSetOrder;
    obj->notify = WaiterNotify;
    memset(obj->order, 0, sizeof(obj->order));
    return obj;
}

客户端

c 复制代码
int main() {
    Barbecuer *boy = InitBarbecuer();
    Waiter *one = InitWaiter();
    // 开放式厨房,可指定厨师@_@
    // 数量为0就表示撤销
    one->setOrder(one, MUTTON, 2, boy);
    one->setOrder(one, MUTTON, 3, boy);
    one->setOrder(one, CHICHKENWING, 4, boy);
    one->notify(one);
    return 0;
}

UML图

总结

  • 命令模式的优点?
    第一,它能较容易地设计一个命令队列;
    第二,在需要的情况下,可以较容易地将命令记入日志;
    第三,允许接收请求的一方决定是否要否决请求;
    第四,可以容易地实现对请求的撤销和重做;
    第五,由于加进新的具体命令类不影响其他的类,因此增加新的具体命令类很容易;

有这些优点的原因是命令模式把请求一个操作的对象与知道怎么执行一个操作的对象分割开。

  • 什么场景使用命令模式?
    命令模式支持撤销/恢复操作功能,但你还不清楚是否需要这个功能时,建议不要使用该模式。敏捷开发原则告诉我们,不要为代码添加基于猜测的、实际不需要的功能。如果不清楚一个系统是否需要命令模式,一般就不要着急去实现它,事实上,在需要的时候通过重构实现这个模式并不困难,只有在真正需要如撤销/恢复操作等功能时,把原来的代码重构为命令模式才有意义。------《大话设计模式》
相关推荐
J_liaty13 小时前
23种设计模式一代理模式
设计模式·代理模式
苏渡苇19 小时前
优雅应对异常,从“try-catch堆砌”到“设计驱动”
java·后端·设计模式·学习方法·责任链模式
短剑重铸之日20 小时前
《设计模式》第十一篇:总结
java·后端·设计模式·总结
feasibility.21 小时前
AI 编程助手进阶指南:从 Claude Code 到 OpenCode 的工程化经验总结
人工智能·经验分享·设计模式·自动化·agi·skills·opencode
BD_Marathon21 小时前
七大设计原则介绍
设计模式
YigAin1 天前
Unity23种设计模式之 享元模式
设计模式·享元模式
不绝1912 天前
UGUI相关——基础篇
命令模式
范纹杉想快点毕业2 天前
实战级ZYNQ中断状态机FIFO设计
java·开发语言·驱动开发·设计模式·架构·mfc
茂桑2 天前
DDD领域驱动设计-基础设施层
设计模式·架构
小温冲冲2 天前
通俗且全面精讲工厂设计模式
设计模式