一. 举例说明

我们知道,在多线程程序中,多个用户都给系统发 Read 和 Write 命令。这里有几点需要说明:

1. 首先明确一点,所有的这些 Read 和 Write 命令都是调用一个库函数。

2. 用户并不需要知道别的用户的存在,也不管别人发不发命令,只管自己发命令,最后给结果即可。

3. 这些命令先是到了一个消息队列里面,然后由消息队列调用库函数。

结构图如下:

代码如下:

  1. class Command;
  2. //实施与执行类
  3. class Reciever
  4. {
  5. public:
  6. void Action()
  7. {
  8. cout<<"Do action !!"<<endl;
  9. }
  10. };
  11. //抽象命令类
  12. class Command
  13. {
  14. public:
  15. virtual ~Command() {}
  16. virtual void Excute() = 0;
  17. protected:
  18. Command() {}
  19. };
  20. //Read 命令
  21. class Read_Command:public Command
  22. {
  23. public:
  24. Read_Command(Reciever* rev)
  25. {
  26. this->_rev = rev;
  27. }
  28. ~Read_Command()
  29. {
  30. delete this->_rev;
  31. }
  32. void Excute()
  33. {
  34. cout<<"Read Command..."<<endl;
  35. _rev->Action();
  36. }
  37. private:
  38. Reciever* _rev;
  39. };
  40. //Write 命令
  41. class Write_Command:public Command
  42. {
  43. public:
  44. Write_Command(Reciever* rev)
  45. {
  46. this->_rev = rev;
  47. }
  48. ~Write_Command()
  49. {
  50. delete this->_rev;
  51. }
  52. void Excute()
  53. {
  54. cout<<"Write_Command..."<<endl;
  55. _rev->Action();
  56. }
  57. private:
  58. Reciever* _rev;
  59. };
  60. //要求命令执行的类
  61. class Invoker
  62. {
  63. public:
  64. Invoker(Command* cmd)
  65. {
  66. _cmd = cmd;
  67. }
  68. Invoker()
  69. {
  70. }
  71. ~Invoker()
  72. {
  73. delete _cmd;
  74. }
  75. //通知执行类执行
  76. void Notify()
  77. {
  78. list<Command*>::iterator it = cmdList.begin();
  79. for (it; it != cmdList.end(); ++it)
  80. {
  81. _cmd = *it;
  82. _cmd->Excute();
  83. }
  84. }
  85. //添加命令
  86. void AddCmd(Command* pcmd)
  87. {
  88. cmdList.push_back(pcmd);
  89. }
  90. //删除命令
  91. void DelCmd(Command* pcmd)
  92. {
  93. cmdList.remove(pcmd);
  94. }
  95. private:
  96. Command* _cmd;
  97. list<Command*> cmdList;
  98. };
  99. //测试代码
  100. int main(int argc,char* argv[])
  101. {
  102. Reciever* rev = new Reciever(); //定义一个执行类
  103. Command* cmd1 = new Read_Command(rev);//Read 命令
  104. Command* cmd2 = new Write_Command(rev);//Write 命令
  105. Invoker inv; //管理所有命令
  106. inv.AddCmd(cmd1);
  107. inv.AddCmd(cmd2);
  108. inv.Notify(); //通知执行类,执行
  109. inv.DelCmd(cmd1);
  110. inv.Notify();
  111. return 0;
  112. }

二. 命令模式

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

优点:

1. 它能比较容易地设计一个命令队列。

2. 在需要的情况下,可以较容易地将命令记入日志。

3. 允许接收请求的一方决定是否要否决请求。

4. 可以容易地实现对请求的撤销和重做。

5. 增加新的具体命令类很容易

6. 把请求一个操作的对象(Command)与知道怎么执行一个操作的对象(Receiver)分割开来。

设计模式C++描述----19.命令(Command)模式的更多相关文章

  1. 命令(Command)模式

    命令模式又称为行动(Action)模式或者交易(Transaction)模式. 命令模式把一个请求或者操作封装到一个对象中.命令模式允许系统使用不同的请求把客户端参数化,对请求排队或者记录请求日志,可 ...

  2. 设计模式的征途—19.命令(Command)模式

    在生活中,我们装修新房的最后几道工序之一是安装插座和开关,通过开关可以控制一些电器的打开和关闭,例如电灯或换气扇.在购买开关时,用户并不知道它将来到底用于控制什么电器,也就是说,开关与电灯.换气扇并无 ...

  3. python 设计模式之命令(Command)模式

    #写在前面 也了解了不少设计模式了,他们都有一个通病,那就是喜欢把简单的东西复杂化.比如在不同的类中加个第三者.哈哈哈,简单变复杂是有目的的,那就是降低耦合度,增强可维护性,提高代码复用性,使代码变得 ...

  4. 十五、命令(Command)模式--行为型模式(Behavioral Pattern)

    命令模式又称为行动(Action)模 式或交易(Transaction)模式.命令模式把一个请求或者操作封装到一个对象中. 命令模式是对命令的封装.命令模式把发出命令的责任和执行命令的责任分割开,委派 ...

  5. 设计模式C++描述----21.解释器(Iterpreter)模式

    一. 解释器模式 定义:给定一个语言,定义它的文法的一种表示,并定一个解释器,这个解释器使用该表示来解释语言中的句子. 结构如下: 代码如下: //包含解释器之外的一些全局信息 class Conte ...

  6. 设计模式C++描述----09.桥接(Bridge)模式

    一. 举例 N年前: 计算机最先出来时,软件和硬件是一绑在一起的,比如IBM出了一台电脑,上面有一个定制的系统,假如叫 IBM_Win,这个IBM_Win系统当然不能在HP电脑上运行,同样HP出的HP ...

  7. 设计模式C++描述----22.访问者(Visitor)模式

    一. 访问者模式 定义:表示一个作用于某对象结构中的各元素的操作.它你可以在不改变各元素的类的前提下定义作用于这些元素的新操作. 结构如下: 二. 举例 假设有一项科学实验,是用来对比两种种子在不同环 ...

  8. 设计模式C++描述----20.迭代器(Iterator)模式

    一. 举例说明 我们知道,在 STL 里提供 Iterator 来遍历 Vector 或者 List 数据结构. Iterator 模式也正是用来解决对一个聚合对象的遍历问题,将对聚合的遍历封装到一个 ...

  9. 设计模式C++描述----17.备忘录(Memento)模式

    一. 备忘录模式 定义:在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态.这样以后就可将该对象恢复到原先保存的状态. 结构图: 使用范围: Memento 模式比较适用于功能 ...

随机推荐

  1. 读《深入理解Elasticsearch》点滴-过滤器

    1.过滤器不影响文档得分 2.过滤的唯一目的是用特定筛选条件来缩小结果范围:而查询不仅缩小结果范围,还会影响文档的得分 3.过滤器运行更加高效(因为不用计算得分) 4.通常过滤器使用Bits接口,返回 ...

  2. 微项目:一步一步带你使用SpringBoot入门(二)

    今天我们来使用JPA做分页项目并且做讲解 如果是新来的朋友请回上一篇 上一篇:微项目(一) maven整合 在pom文件的dependencies依赖中导入以下依赖 <dependency> ...

  3. 【SQL server初级】SQL SERVER Transactional Replication中添加新表如何不初始化整个快照

    在SQL SERVER的复制(Replication)中,有可能出现由于业务需求变更,需要新增一张表或一些表到已有的复制(发布订阅)当中,这种需求应该是很正常,也很常见的.但是在已有的复制(发布订阅) ...

  4. 【SQL server初级】数据库性能优化三:程序操作优化

    数据库优化包含以下三部分,数据库自身的优化,数据库表优化,程序操作优化.此文为第三部分 数据库性能优化三:程序操作优化 概述:程序访问优化也可以认为是访问SQL语句的优化,一个好的SQL语句是可以减少 ...

  5. 说说 Java 线程间通信

    序言 正文 一.Java线程间如何通信? 线程间通信的目标是使线程间能够互相发送信号,包括如下几种方式: 1.通过共享对象通信 线程间发送信号的一个简单方式是在共享对象的变量里设置信号值:线程A在一个 ...

  6. Shell之Function与Source

    目录 Shell之Function与Source 参考 Fuction的编写 Source的使用 Shell之Function与Source

  7. chown、chgrp、chmod

    1.权限对应的数值 一开始理解权限对应的数值总是要去用二进制去算例如r-x是多少rwx是多少,后来才知道r就 是4,w就是2,x就是1,不管权限怎么变,他们对应的数值就是对应位相加.. 权限对于文件来 ...

  8. mysql基础操作 增删改查

    如何使用终端操作数据库 如何登录数据库 mysql -u用户名 -p密码 比如: mysql -uroot -p123456 如何查询数据库服务器中所有的数据库 show databases; 如何选 ...

  9. Object.keys方法详解

    一.官方解释 Object.keys() 方法会返回一个由一个给定对象的自身可枚举属性组成的数组,数组中属性名的排列顺序和使用 for...in 循环遍历该对象时返回的顺序一致 .如果对象的键-值都不 ...

  10. CentOS8 yum/dnf 配置国内源

    CentOS8 yum/dnf 配置国内源(临时) CentOS 8更改了软件包的安装程序,取消了 yum 的配置方法,改而使用了dnf 作为安装程序.虽然改变了软件包的安装方式,但是 dnf 还是能 ...