设计模式二十四章经之命令模式

概述

命令模式相对于其他模式没有那么多的规定,因为它不是一个很规矩的模式。不过,正因为这一点,命令模式对比其他设计模式更为灵活。我们接触比较多的命令模式无非开关机。我们点击关机命令时,它会先暂停事件,然后结束所有进程,最后关机。也就是说我只需要点击一个关机按钮就可以做到这一系列的命令。而我们的命令模式其实也和这一样。讲一系列的动作进行请求封装,用户只需要调用一个方法,这些动作会被挨个执行。

使用场景

需要抽象出待执行的动作,然后以参数的形式提供出来。类似于过程设计中的回调机制,而命令模式正是回调机制的一个面向对象的代替品。

在不同的时刻指定、排列和执行请求。一个命令对象可以有与初始请求无关的生存期。

需要支持取消操作。

需要支持事务操作。

支持修改日志,这样当系统崩溃时,这些修改可以被重做一遍。

具体实现

假设我们现在关机,它会先后关闭浏览器,音乐和QQ,最后才会关机。现在我们来模拟一下。

public class CloseReceiver{
    public void action(){
        System.out.println("关机");
    }

    public void closeQQ(){
        System.out.println("关闭QQ");
        action();
    }

    public void closeMusic(){
        System.out.println("关闭音乐");
    }

    public void closeChrome(){
        System.out.println("关闭浏览器");
    }
}
public interface Close{
    void execute();
}
public class QQClose implements Close{
    private CloseReceiver receiver;

    public QQClose(CloseReceiver receiver){
        this.receiver=receiver;
    }

    @Override
    public void excute(){
        recevier.closeQQ();
    }
}
public class MusicClose implements Close{
    private CloseReceiver receiver;

    public MusicClose(CloseReceiver receiver){
        this.receiver=receiver;
    }

    @Override
    public void excute(){
        recevier.closeMusic();
    }
}
public class ChromeClose implements Close{
    private CloseReceiver receiver;

    public ChromeClose(CloseReceiver receiver){
        this.receiver=receiver;
    }

    @Override
    public void excute(){
        recevier.closeChrome();
    }
}
public class Invoker{
    private QQClose qqclose;
    private MusicClose musicclose;
    private ChromeClose chromeclose;

    public setQQClose(QQClose qqclose){
        this.qqclose=qqclose;
    } 

    public setMusicClose(MusicClose musicclose){
        this.musicclose=musicclose;
    }  

    public setChromeClose(ChromeClose chromeclose){
        this.chromeclose=chromeclose;
    }

    public void closeChrome(){
        chromeclose.excute();
    } 

    public void closeMusic(){
        musicclose.excute();
    }  

    public void closeQQ(){
        qqclose.excute();
    }
}
public void Client{

    public static void main(String[] args){
        CloseReceiver receiver=new CloseReceiver();
        QQClose qqclose=new QQClose(receiver);
        MusicClose musicclose=new MusicClose(receiver);
        ChromeClose chromeclose=new ChromeClose(receiver);
        Invoker invoker=new Invoker();
        invoker.closeChrome();
        invoker.closeMusic();
        invoker.closeQQ();
    }
}

最后输出:

关闭浏览器
关闭音乐
关闭QQ
关机

调用逻辑做的如此复杂是因为开发起来方便。每当增加或者修改关闭软件时,我们只需要处理CloseReceiver以及它对应的Close类即可。设计模式原则中有一个重要的原则,开闭原则。大家可以思考思考。

除此之外,命令模式可以实现记录命令的功能。如果我们在Invoker中用一个数据结构来存储这些执行的命令,以此可以知道刚才执行过什么命令,并可以在需要时恢复。

总结

在命令模式中,可以充分体验出设计模式的通病。类的膨胀。但其也有相应的优点。比如降低了系统的耦合度,新的命令可以添加到系统中去等等。

原文发布于微信公众号 - 我就是马云飞(coding_ma)

原文发表时间:2018-06-12

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏用户2442861的专栏

高性能网络编程4--TCP连接的关闭

TCP连接的关闭有两个方法close和shutdown,这篇文章将尽量精简的说明它们分别做了些什么。

772
来自专栏Java技术

使用Spring AOP实现MySQL数据库读写分离案例分析

分布式环境下数据库的读写分离策略是解决数据库读写性能瓶颈的一个关键解决方案,更是最大限度了提高了应用中读取 (Read)数据的速度和并发量。

702
来自专栏云霄雨霁

数据链路层协议ARQ

1050
来自专栏我是攻城师

如何管理Spark Streaming消费Kafka的偏移量(三)

3696
来自专栏Java职业技术分享

java框架之Spring 核心框架体系结构

很多人都在用spring开发java项目,但是配置maven依赖的时候并不能明确要配置哪些spring的jar,经常是胡乱添加一堆,编译或运行报错就继续配置ja...

480
来自专栏Hadoop实操

如何在CM中启用YARN的使用率报告

CDH的高级功能"群集利用率报告"(Cluster Utilization Report)是整个多租户方案体系里的一部分,可以用来查看租户的资源使用情况,并可以...

4005
来自专栏zhangdd.com

zabbix监控-清理zabbix 历史数据

zabbix运行一段时间之后,会留下大量的历史 数据,会发现zabbix的数据库一直在增大。运行3个月后笔者的数据库达到了5.7G,可能造成系统性能下降,查看历...

841
来自专栏虚拟化云计算

libvirt-内存分配和内存热插拔

在qemu命令中使用参数-m来设置的。表示虚拟机在启动阶段使用的内存。包括在启动或稍后热插拔时指定的可能的附加内存。

3799
来自专栏Java3y

Struts2【整合Spring】

前言 Spring的第二和第三篇已经讲解了Spring的基本要点了【也就是Core模块】…本博文主要讲解Spring怎么与Struts2框架整合… Struts...

3389
来自专栏DevOps时代的专栏

如何使用 Jenkins 的脚本化流水线(Pipeline)

在这篇简单的教程中,你将会学习到 Jenkins 的流水线即代码,以及如何开发流水线脚本的指导。 Jenkins 是一个开源持续集成服务器,它可以提供持续执行自...

6287

扫码关注云+社区