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图

总结

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

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

  • 什么场景使用命令模式?
    命令模式支持撤销/恢复操作功能,但你还不清楚是否需要这个功能时,建议不要使用该模式。敏捷开发原则告诉我们,不要为代码添加基于猜测的、实际不需要的功能。如果不清楚一个系统是否需要命令模式,一般就不要着急去实现它,事实上,在需要的时候通过重构实现这个模式并不困难,只有在真正需要如撤销/恢复操作等功能时,把原来的代码重构为命令模式才有意义。------《大话设计模式》
相关推荐
茶本无香2 小时前
设计模式之六—组合模式:构建树形结构的艺术
java·设计模式·组合模式
橘色的喵2 小时前
嵌入式 C++ 高性能流式架构的设计
数码相机·设计模式
会员果汁3 小时前
20.设计模式-职责链模式
设计模式
羊群智妍12 小时前
2026GEO监测工具精选:免费AI搜索优化监测工具实用指南
百度·微信·命令模式·新浪微博·segmentfault
她和夏天一样热17 小时前
【设计模式】工厂方法模式在开发中的真实应用
设计模式·工厂方法模式
烤麻辣烫1 天前
23种设计模式(新手)-9单例模式
java·开发语言·学习·设计模式·intellij-idea
资生算法程序员_畅想家_剑魔1 天前
Java常见技术分享-设计模式的六大原则
java·开发语言·设计模式
刀法如飞1 天前
从零手搓一个类Spring框架,彻底搞懂Spring核心原理
java·设计模式·架构设计
一条闲鱼_mytube1 天前
智能体设计模式全景总结:21个模式快速串联指南
网络·设计模式