命令模式

元素

  • 抽象命令/具体命令
  • 抽象接受者/具体接受者
  • 调用者
  • 客户端
command_url.png

应用场景举例

这里我们用一个场景来描述:去川湘阁饭店点一份剁椒鱼头和宫保鸡丁。
这里我们把整个关键流程写出来

  • 点菜下单
  • 收银台出单
  • 厨房收到抄菜单,分给具体厨师
  • 厨师炒菜

接下来我们就可以抽象出具体的角色。客户端当然就是我们的用户,调用者就是收银台出单的妹纸,命令就是抄菜单,接受者就是厨师。这样就很清晰了,客户端(用户)不需要知道功能(炒菜)怎么实现或者谁实现,他只需要找调用者(服务员/前台)描述清楚,然后调用者发命令给接受者(厨师),然后接受者执行命令(炒菜)。 这个流程非常的清晰也不会出错。什么样的命令给什么样的接受者执行,比如:剁椒鱼头,就打剁椒鱼头的票,然后指定给会做剁椒鱼头的师傅做。宫保鸡丁就给会做宫保鸡丁的师傅做。这个过程中,命令(小票)包含接受者(师傅)的信息。不同的命令对应不同的接受者。所以这里建立命令和接受者的抽象。

优点

  • 将请求的发起者和执行者接口,通过命令来实现,将客户端的调用参数化。只需要将每个动作封装正命令,由发起者命令执行者来执行
  • 请求排队、记录每个请求。拿上面的场景来说,当很多客人点了剁椒鱼头时,厨师可能做不过来,这时候就得排队,先来的先做。

示例(C++版):

接受者(厨师)抽象类
#include<iostream>
using namespace std;
class Chef
{
public:
    virtual int cooking(int id)=0;
    virtual void printTicketIds()const = 0;
    virtual ~Chef() {}
};
具体接受者(川湘阁做剁椒鱼头的厨师)类
#include "Chef.h"
class FishChef :
    public Chef
{
    int id;
    string name;
    deque<int> ticketIds;
public:
    FishChef(int id, string name);
    int cooking(int id);
    void printTicketIds()const;
    ~FishChef();
};
#include "FishChef.h"


FishChef::FishChef(int id, string name)
{
    this->id = id;
    this->name = name;
}


FishChef::~FishChef()
{
}

void FishChef::printTicketIds()const {
    cout << "id为" << this->id << "  姓名为:" << this->name.c_str() << "的厨师需要做的菜";
    for (deque<int>::const_iterator it = ticketIds.begin(); it != ticketIds.end(); it++) {
        cout << *it << " ";
    }
    cout << endl;
}

int FishChef::cooking(int id) {
    if (ticketIds.size() > 3) {
        return 1;
    }
    ticketIds.push_back(id);
    return 0;
}
抽象命令(炒菜单)
#include<iostream>
#include "Chef.h"
using namespace std;
class Order
{
protected :
    Chef* chef;
public:
    Order(Chef* chef) {
        this->chef = chef;
    }
    virtual void make()=0;
    ~Order() {
        if (chef != NULL)
            delete chef;
    }
};
具体命令(川湘阁的抄菜单)
#include "stdafx.h"
#include "ChuangXiangGeOrder.h"

ChuangXiangGeOrder::ChuangXiangGeOrder(Chef * chef, int id):Order(chef)
{
    this->id = id;
}

void ChuangXiangGeOrder::make()
{
    if (chef == NULL) {
        cout << "请先设置厨师" << endl;
    }
    int res = chef->cooking(id);
}

ChuangXiangGeOrder::~ChuangXiangGeOrder()
{
}
抽象调用者(下单服务员)
#include<iostream>
#include<vector>
#include<string>
#include"Order.h"
using namespace std;
using namespace std;
class Waiter
{
protected:
    Order* order;
public:
    Waiter() {}
    virtual pair<int,string> setOrder(Order* order)=0;
    virtual void execute()=0;
    ~Waiter() {}
};
具体调用者(川湘阁的下单服务员)
#include "stdafx.h"
#include "ChuanXiangGeWaiter.h"

ChuanXiangGeWaiter::ChuanXiangGeWaiter()
{
}

pair<int, string> ChuanXiangGeWaiter::setOrder(Order* order)
{
    Waiter::order = order;
    return pair<int, string>();
}

void ChuanXiangGeWaiter::execute()
{
    if (order == NULL) {
        cout << "请先下单" << endl;
    }
    order->make();
}


ChuanXiangGeWaiter::~ChuanXiangGeWaiter()
{
    
}
测试代码
int main()
{
    Chef *chef = new FishChef(1, "张厨师");
    Order *order = new ChuangXiangGeOrder(chef,1001);
    Waiter *waiter = new ChuanXiangGeWaiter();
    waiter->setOrder(order);
    waiter->execute();
    chef->printTicketIds();

    Order *order1 = new ChuangXiangGeOrder(chef, 1002);
    waiter->setOrder(order1);
    waiter->execute();
    chef->printTicketIds();

    Order *order2 = new ChuangXiangGeOrder(chef, 1003);
    waiter->setOrder(order2);
    waiter->execute();
    chef->printTicketIds();
    return 0;
}

结果:

command.png

java版

线程池
Executor就是抽象调用者

public interface Executor {

    /**
     * Executes the given command at some time in the future.  The command
     * may execute in a new thread, in a pooled thread, or in the calling
     * thread, at the discretion of the {@code Executor} implementation.
     *
     * @param command the runnable task
     * @throws RejectedExecutionException if this task cannot be
     * accepted for execution
     * @throws NullPointerException if command is null
     */
    void execute(Runnable command);
}

我们看到其中一个实现类ThreadPoolExecutor


图片来自官方api文档

看到Executor的源码,我们看到参数的起名都是command。这里Runable就是抽象的命令。那接受者呢。其实这里的接受者也是Runable。同时ThreadPoolExecutor除了能够发起命令外,还可以对Runable进行排队。这里我们用一个例子说明

测试
public class TestMain {
    public static void main(String[] args) {
        new Thread(new TaskA(1)).start();
        ExecutorService executor = Executors.newFixedThreadPool(2);
        executor.execute(new TaskA(2));
        executor.execute(new TaskA(3));
        executor.execute(new TaskA(4));
    }

    static class TaskA implements Runnable {
        private int id;

        public TaskA(int id) {
            this.id = id;
        }

        public void run() {
            log(id+" TaskA begin");
            try {
                Thread.sleep(2000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            log(id+" TaskA end");
        }
    }
    public static void log(String content) {
        System.out.println(content);
    }
}

结果

1 TaskA begin
2 TaskA begin
3 TaskA begin
1 TaskA end
3 TaskA end
2 TaskA end
4 TaskA begin
4 TaskA end
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容