一. 访问者模式

定义:表示一个作用于某对象结构中的各元素的操作。它你可以在不改变各元素的类的前提下定义作用于这些元素的新操作。

结构如下:

二. 举例

假设有一项科学实验,是用来对比两种种子在不同环境下的生长情况。

两种种子,一种是普通的种子(Seed_A),一种是太空运回的种子(Seed_B)。

生长环境,分别是在多雨环境下(Rain_Status),阳光环境下(Sun_Status)等等。

结构如下:

代码如下:

  1. //状态
  2. class Status
  3. {
  4. public:
  5. virtual ~Status() {}
  6. virtual void VisitSeed_A(Seed* elm) {}
  7. virtual void VisitSeed_B(Seed* elm) {}
  8. protected:
  9. Status() {}
  10. };
  11. //下雨状态
  12. class Rain_Status:public Status
  13. {
  14. public:
  15. Rain_Status() {}
  16. virtual ~Rain_Status() {}
  17. //下雨状态下A种子生长的情况
  18. virtual void VisitSeed_A(Seed* elm)
  19. {
  20. cout<<"Rain will visit Seed A..."<<endl;
  21. }
  22. //下雨状态下B种子生长的情况
  23. virtual void VisitSeed_B(Seed* elm)
  24. {
  25. cout<<"Rain will visit Seed B..."<<endl;
  26. }
  27. };
  28. //阳光状态
  29. class Sun_Status:public Status
  30. {
  31. public:
  32. Sun_Status() {}
  33. virtual ~Sun_Status() {}
  34. //阳光状态下A种子生长的情况
  35. virtual void VisitSeed_A(Seed* elm)
  36. {
  37. cout<<"Sun will visit Seed A..."<<endl;
  38. }
  39. //阳光状态下B种子生长的情况
  40. virtual void VisitSeed_B(Seed* elm)
  41. {
  42. cout<<"Sun will visit Seed B..."<<endl;
  43. }
  44. };
  45. //种子
  46. class Seed
  47. {
  48. public:
  49. virtual ~Seed() {}
  50. virtual void Accept(Status* vis) = 0;
  51. protected:
  52. Seed() {}
  53. };
  54. //种子A,假设为普通种子
  55. class Seed_A:public Seed
  56. {
  57. public:
  58. Seed_A() {}
  59. ~Seed_A() {}
  60. void Accept(Status* vis)
  61. {
  62. vis->VisitSeed_A(this);
  63. }
  64. };
  65. //种子B,假设为从太空带回来的种子
  66. class Seed_B:public Seed
  67. {
  68. public:
  69. Seed_B() {}
  70. ~Seed_B() {}
  71. void Accept(Status* vis)
  72. {
  73. vis->VisitSeed_B(this);
  74. }
  75. };
  76. //对象结构类,为了对比不同种子
  77. class ObjectStructure
  78. {
  79. private:
  80. list<Seed*> lseed;
  81. public:
  82. //Add
  83. void Attach(Seed* seed)
  84. {
  85. lseed.push_back(seed);
  86. }
  87. //Delete
  88. void Detach(Seed* seed)
  89. {
  90. lseed.remove(seed);
  91. }
  92. //Show
  93. void Display(Status* status)
  94. {
  95. list<Seed*>::iterator it = lseed.begin();
  96. for (it; it != lseed.end(); ++it)
  97. {
  98. (*it)->Accept(status);
  99. }
  100. }
  101. };
  102. //测试代码
  103. int main(int argc,char* argv[])
  104. {
  105. ObjectStructure obj;
  106. //加入要对比的两个种子
  107. obj.Attach(new Seed_A());
  108. obj.Attach(new Seed_B());
  109. //查看各种状态下两个种子的情况
  110. obj.Display(new Rain_Status());
  111. //Sun Satte
  112. obj.Display(new Sun_Status());
  113. return 0;
  114. }

三. 说明

1. 首先有一点要明确,就是两种种子不会轻易改变,也就是只有普通和太空种子两种。换句话说就是,数据结构比较稳定。

2. 可以变的是新增的状态,比如增加一个X光下的生成情况,等等。说白了就是,操作集合可以相对自由的演化。

3. 这种结构的优点是,增加新的操作很容易;缺点是,增加新的数据结构有点困难,因为你要在每一个访问者里都添加相应的操作。

4. 种子生长图相对于访问者模式的结构图有如下关系:

seed(种子)相当于 element(元素),这个是不怎么变的。

status(状态) 相当于 visitor(访问者),这个是可变且易变的。要注意的是,每个访问者都要对所有的元素(element)进行操作。

5. 事实上我们很少用这种模式,因为数据结构(element)不变的情况很少。

设计模式C++描述----22.访问者(Visitor)模式的更多相关文章

  1. 设计模式之第22章-组合模式(Java实现)

    设计模式之第22章-组合模式(Java实现) “鱼哥,有没有什么模式是用来处理树形的“部分与整体”的层次结构的啊.”“当然”“没有?”“有啊.别急,一会人就到了.” 组合模式之自我介绍 “请问你是?怎 ...

  2. 《图解设计模式》读书笔记6-1 VISITOR模式

    目录 1. Visitor模式简介 2. 示例 2.1 类图 2.2 代码 3. 模式的角色和类图 4. 思路拓展 4.1 双重分发 4.2 开闭原则 4.3 难以增加ConcreteElement角 ...

  3. 设计模式-访问者(Visitor)模式

    访问者模式是对象的行为模式.访问者模式的目的是封装施加在某种数据结构元素上的操作.一旦一些操作需要修改,接受这个操作的数据结构可以保持不变. 个人觉得访问者模式相对其他的设计模式来说稍微复杂,难理解一 ...

  4. [设计模式]访问者 Visitor 模式

    访问者模式是对象的行为模式. 访问者模式的目的是封装一些施加于某种数据结构元素之上的操作.一旦这些操作需要修改的话,接受这个操作的数据结构则可以保持不变.

  5. C++设计模式实现--訪问者(Visitor)模式

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/L_Andy/article/details/36896645 一. 訪问者模式 定义:表示一个作用于 ...

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

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

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

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

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

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

  9. 设计模式C++描述----19.命令(Command)模式

    一. 举例说明 我们知道,在多线程程序中,多个用户都给系统发 Read 和 Write 命令.这里有几点需要说明: 1. 首先明确一点,所有的这些 Read 和 Write 命令都是调用一个库函数. ...

随机推荐

  1. helm部署Filebeat + ELK

    helm部署Filebeat + ELK 系统架构图: 1) 多个Filebeat在各个Node进行日志采集,然后上传至Logstash 2) 多个Logstash节点并行(负载均衡,不作为集群),对 ...

  2. 【ADO.NET基础】后台获取前台控件

    C# 后台获取前台 input 文本框值.string aa=Request.Form[headself]; 那么要是后台给前台input文本框赋值呢? 后台 public string Headse ...

  3. Jquery 动态添加元素 添加点击事件

    给动态添加的元素添加js事件,不能直接添加js事件,需要借助on( )方法,给选择器指定的子元素添加事件处理函数,并非选择器本身: 1. 语法 $(selector).on(event,childSe ...

  4. .NET Core 3.0 可卸载程序集原理简析

    因为最近在群里被问到如何理解 .NET Core 3.0 可卸载程序集,所以就写了这篇简单的分析. 因为时间实在很少,这篇文章只简单的罗列了相关的代码,请配合官方说明文档理解. 另外,书籍<.N ...

  5. 瀑布流实例及懒加载(echo.js)

    瀑布流布局: 图片等宽,不定高,按最低高度来顺序排列:实现方法:获取每次获取四行中最低高度对应的一行,将下一张加载的图片放在该位置,每次加载前都获取最低高度: ①请求图片的接口    地址此php文件 ...

  6. RocketMQ 源码学习笔记————Producer 是怎么将消息发送至 Broker 的?

    目录 RocketMQ 源码学习笔记----Producer 是怎么将消息发送至 Broker 的? 前言 项目结构 rocketmq-client 模块 DefaultMQProducerTest ...

  7. SpringBoot注入Mapper失败

    SpringBoot注入Mapper失败,可能是因为没有加扫描Mapper层的注解 方式一:在所有mapper接口使用@Mapper注解 @Mapper public interface UserMa ...

  8. jsonp与cors跨域解析

    1.浏览器的同源安全策略 没错,就是这家伙干的,浏览器只允许请求当前域的资源,而对其他域的资源表示不信任.那怎么才算跨域呢? 请求协议http,https的不同 域domain的不同 端口port的不 ...

  9. Cocos Creator 通用框架设计 —— 资源管理

    如果你想使用Cocos Creator制作一些规模稍大的游戏,那么资源管理是必须解决的问题,随着游戏的进行,你可能会发现游戏的内存占用只升不降,哪怕你当前只用到了极少的资源,并且有使用cc.loade ...

  10. 你也可以写个服务器 - C# Socket学习2

    前言 这里说的服务器是Web服务器,是类似IIS.Tomcat之类的,用来响应浏览器请求的服务. Socket模拟浏览器的Url Get请求 首先浏览器的请求是HTTP协议.我们上一篇说过,HTTP是 ...