Java设计模式百例 - 命令模式

本文源码见:https://github.com/get-set/get-designpatterns/tree/master/command

命令模式(Command Pattern)是一种数据驱动的设计模式,它属于行为型模式。请求以命令的形式包裹在对象中,并传给调用对象。调用对象寻找可以处理该命令的合适的处理对象,并把该命令传给这个处理对象,该处理对象执行命令。

命令模式把发出命令的责任和执行命令的责任分开,委派个不同的对象。什么情况下需要吧发出命令的责任和执行命令的责任分开呢?不好意思,按照我的尿性,又要搬出画图的例子了。

例子

我们先考虑发出命令和执行命令耦合起来的情况,看有啥不好。比如画图有如下几个命令:画形状、填充颜色、组合形状等,每个操作命令的执行由不同的对象来实现。当用户在执行不同的操作的时候,画图软件有个统一的逻辑来处理:

// 以下逻辑位于“命令发出者”的方法内:
if(要画形状){
    形状处理对象.画形状("圆形");
}
if(要填充颜色){
    颜色处理对象.填充颜色("红色");
}
if(要组合形状){
    组合处理对象.组合形状([]{"红色圆形","蓝色矩形","黄色三角形");
}

这里有几个不太灵活的地方:

  1. “命令发出者”要同所有的“命令执行者”维持关系(前者要有后者的所有引用),只要增加一种命令执行者,就需要修改命令发出者的代码逻辑,显然不符合“开闭”原则的吧;尤其是命令发出者如果是不开放修改的(比如位于框架中),就很难保证扩展性了;
  2. 命令执行有可能失败,或者Ctrl+Z回退,这种操作显然是经常遇到的,对于”命令发出者“的逻辑复杂度来说,显然又是雪上加霜。
before-decoupling.png

我们再来看一下,应用命令模式如何解耦以及解耦后有什么好处。

将发出命令和执行命令解耦,即“发出命令者”不直接调用“执行命令者”的方法,而是通过一个“中间媒介”,这个“中间媒介”就是“命令”。从而由原来的“发出命令者 - 执行命令者”的关系变成“发出命令者 - 命令 - 执行命令者”

“发出命令者”只关心命令的执行,并不关心由谁来执行。从而它手中不再握有一系列的“执行命令者”的引用,而是只有处理命令(或命令集合,如命令队列)的引用。而命令本身有“执行命令者”的引用,从而知道找谁去完成命令的具体执行。

after-decoupling.png

当然,要做到完全解耦和保证灵活性,必须针对“命令”进行面向接口的设计,也就是“发出命令者”手里的只有抽象类型的命令对象,才能做到“不care”。这都是设计模式的老掉牙的套路了,不再多说。

如此,刚才提到的两个不太灵活的问题也就迎刃而解了。看代码喽~

发出命令者 DrawingApp.java

public class DrawingApp {
    private List<Command> commands = new ArrayList<Command>();
    public void takeCommand(Command command) {
        commands.add(command);
    }
    public void CommandsDone() {
        for (Command command:commands) {
            command.doCmd();
        }
    }
    public void undoLastCommand() {
        commands.get(commands.size() - 1).undoCmd();
    }
}

看到它引用了命令的列表。

命令抽象 Command.java

public interface Command {
    void doCmd();
    void undoCmd();
}

具体命令 ShapeDrawing.java

public class ShapeDrawing implements Command {
    private ShapeDrawer drawer;
    private String arg;

    public ShapeDrawing(ShapeDrawer drawer, String arg) {
        this.drawer = drawer;
        this.arg = arg;
    }

    public void doCmd() {
        drawer.drawShape(arg);
    }

    public void undoCmd() {
        drawer.undrawShape();
    }
}

具体命令 ColorFilling.java

public class ColorFilling implements Command {
    private ColorFiller filler;
    private String arg;

    public ColorFilling(ColorFiller filler, String arg) {
        this.filler = filler;
        this.arg = arg;
    }

    public void doCmd() {
        filler.fillColor(arg);
    }

    public void undoCmd() {
        filler.unfillColor();
    }
}

两个具体命令都引用了相应的命令执行者。

执行命令者 ShapeDrawer.java

public class ShapeDrawer {
    public void drawShape(String shape) {
        System.out.println("画了一个" + shape);
    }
    public void undrawShape() {
        System.out.println("撤销刚才画的形状");
    }
}

执行命令者 ColorFiller.java

public class ColorFiller {
    public void fillColor(String color) {
        System.out.println("填充" + color);
    }
    public void unfillColor() {
        System.out.println("撤销填充的颜色");
    }
}

测试一下,发出三个命令,并撤销最后一个:

Client.java

public class Client {
    public static void main(String[] args) {
        DrawingApp app = new DrawingApp();
        ShapeDrawer shapeDrawer = new ShapeDrawer();
        ColorFiller colorFiller = new ColorFiller();
        Command drawCircle = new ShapeDrawing(shapeDrawer, "圆形");
        Command fillRed = new ColorFilling(colorFiller, "红色");
        Command drawRectancle = new ShapeDrawing(shapeDrawer, "矩形");
        app.takeCommand(drawCircle);
        app.takeCommand(fillRed);
        app.takeCommand(drawRectancle);
        app.CommandsDone();
        app.undoLastCommand();
    }
}

输出如下:

画了一个圆形
填充红色
画了一个矩形
撤销刚才画的形状

总结

通过以上例子,我们可以总结出命令模式的特点:

  1. 将命令封装为对象作为传递媒介;
  2. 命令发出者(invoker)和命令执行者(receiver)是解耦的,通过命令对象“牵线”,invoker维护有命令(或命令集合)的引用,命令中有receiver的引用。

命令模式的应用场景(转载):

  1. Multi-level undo(多级undo操作)。如果系统需要实现多级回退操作,这时如果所有用户的操作都以command对象的形式实现,系统可以简单地用stack来保存最近执行的命令,如果用户需要执行undo操作,系统只需简单地popup一个最近的command对象然后执行它的undo()方法既可。
  2. Transactional behavior(原子事务行为)。借助command模式,可以简单地实现一个具有原子事务的行为。当一个事务失败时,往往需要回退到执行前的状态,可以借助command对象保存这种状态,简单地处理回退操作。
  3. Progress bars(状态条)。假如系统需要按顺序执行一系列的命令操作,如果每个command对象都提供一个getEstimatedDuration()方法,那么系统可以简单地评估执行状态并显示出合适的状态条。
  4. Wizards(导航)。通常一个使用多个wizard页面来共同完成一个简单动作。一个自然的方法是使用一个command对象来封装wizard过程,该command对象在第一个wizard页面显示时被创建,每个wizard页面接收用户输入并设置到该command对象中,当最后一个wizard页面用户按下“Finish”按钮时,可以简单地触发一个事件调用execute()方法执行整个动作。通过这种方法,command类不包含任何跟用户界面有关的代码,可以分离用户界面与具体的处理逻辑。
  5. GUI buttons and menu items(GUI按钮与菜单条等等)。Swing系统里,用户可以通过工具条按钮,菜单按钮执行命令,可以用command对象来封装命令的执行。
  6. Thread pools(线程池)。通常一个典型的线程池实现类可能有一个名为addTask()的public方法,用来添加一项工作任务到任务队列中。该任务队列中的所有任务可以用command对象来封装,通常这些command对象会实现一个通用的接口比如java.lang.Runnable。
  7. Macro recording(宏纪录)。可以用command对象来封装用户的一个操作,这样系统可以简单通过队列保存一系列的command对象的状态就可以记录用户的连续操作。这样通过执行队列中的command对象,就可以完成"Play back"操作了。
  8. Networking。通过网络发送command命令到其他机器上运行。
  9. Parallel Processing(并发处理)。当一个调用共享某个资源并被多个线程并发处理时。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 198,322评论 5 465
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,288评论 2 375
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 145,227评论 0 327
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,015评论 1 268
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,936评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 47,534评论 1 275
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,995评论 3 389
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,616评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,907评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,923评论 2 315
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,741评论 1 328
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,525评论 3 316
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,016评论 3 301
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,141评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,453评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,054评论 2 343
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,249评论 2 339

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,524评论 18 399
  • 1 场景问题# 1.1 如何开机## 估计有些朋友看到这个标题会非常奇怪,电脑装配好了,如何开机?不就是按下启动按...
    七寸知架构阅读 2,784评论 1 59
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,530评论 18 139
  • 版本说明 os:CentOS 6.8 php:5.5.38 nginx:1.10.3 mysql:5.6 安装步骤...
    wangtingkui阅读 363评论 0 2
  • 发现美的眼睛,用镜头留住美的瞬间! 说说我爱上手机微距摄影的八个理由。 一、摄影让你关注细节 当你真正喜欢上摄影,...
    小盒子碎碎念阅读 2,079评论 9 15