学习php设计模式 php实现访问者模式(Visitor)


Posted in PHP onDecember 07, 2015

访问者模式表示一个作用于某对象结构中各元素的操作。它可以在不修改各元素类的前提下定义作用于这些元素的新操作,即动态的增加具体访问者角色。
访问者模式利用了双重分派。先将访问者传入元素对象的Accept方法中,然后元素对象再将自己传入访问者,之后访问者执行元素的相应方法。
访问者模式多用在聚集类型多样的情况下。在普通的形式下必须判断每个元素是属于什么类型然后进行相应的操作,从而诞生出冗长的条件转移语句。而访问者模式则可以比较好的解决这个问题。对每个元素统一调用$element->accept($vistor)即可。
访问者模式多用于被访问的类结构比较稳定的情况下,即不会随便添加子类。访问者模式允许被访问结构添加新的方法。
Visitor模式实际上是分离了对象结构中的元素和对这些元素进行操作的行为,从而使我们在根据对象结构中的元素进行方法调用的时候,不需要使用IF语句判断,也就是封装了操作。
但是,如果增加新的元素节点,则会导致包括访问者接口及其子类的改变,这就会违反了面向对象中的开闭原则
当这种情况出现时,一般表示访问者模式已经可能不再适用了,或者说设计时就有问题了!
一、Visitor模式结构图

学习php设计模式 php实现访问者模式(Visitor)

二、Visitor模式中主要角色

1)抽象访问者角色(Visitor):为该对象结构(ObjectStructure)中的每一个具体元素提供一个访问操作接口。该操作接口的名字和参数标识了 要访问的具体元素角色。这样访问者就可以通过该元素角色的特定接口直接访问它。
2)具体访问者角色(ConcreteVisitor):实现抽象访问者角色接口中针对各个具体元素角色声明的操作。
3)抽象节点(Node)角色:该接口定义一个accept操作接受具体的访问者。
4)具体节点(Node)角色:实现抽象节点角色中的accept操作。
5) 对象结构角色(ObjectStructure):这是使用访问者模式必备的角色。它要具备以下特征:能枚举它的元素;可以提供一个高层的接口以允许该访问者访问它的元素;可以是一个复合(组合模式)或是一个集合,如一个列表或一个无序集合(在PHP中我们使用数组代替,因为PHP中的数组本来就是一个可以放置任何类型数据的集合)
三、Visitor模式的优缺点
访问者模式有如下的优点
1)访问者模式使得增加新的操作变得很容易。使用访问者模式可以在不用修改具体元素类的情况下增加新的操作。它主要是通过元素类的accept方法来接受一个新的visitor对象来实现的。如果一些操作依赖于一个复杂的结构对象的话,那么一般而言,增加新的操作会很复杂。而使用访问者模式,增加新的操作就意味着增加一个新的访问者类,因此,变得很容易。
2)访问者模式将有关的行为集中到一个访问者对象中,而不是分散到一个个的节点类中。
3)访问者模式可以跨过几个类的等级结构访问属于不同的等级结构的成员类。迭代子只能访问属于同一个类型等级结构的成员对象,而不能访问属于不同等级结构的对象。访问者模式可以做到这一点。
4)积累状态。每一个单独的访问者对象都集中了相关的行为,从而也就可以在访问的过程中将执行操作的状态积累在自己内部,而不是分散到很多的节点对象中。这是有益于系统维护的优点。

访问者模式有如下的缺点
1)增加新的节点类变得很困难。每增加一个新的节点都意味着要在抽象访问者角色中增加一个新的抽象操作,并在每一个具体访问者类中增加相应的具体操作。
2)破坏封装。访问者模式要求访问者对象访问并调用每一个节点对象的操作,这隐含了一个对所有节点对象的要求:它们必须暴露一些自己的操作和内部状态。不然,访问者的访问就变得没有意义。由于访问者对象自己会积累访问操作所需的状态,从而使这些状态不再存储在节点对象中,这也是破坏封装的。

使用Visitor模式的前提: 对象群结构中(Collection) 中的对象类型很少改变。
在接口Visitor和Element中,确保Element很少变化,也就是说,确保不能频繁的添加新的Element元素类型加进来,可以变化的是访问者行为或操作,也就是Visitor的不同子类可以有多种,这样使用访问者模式最方便.
如果对象集合中的对象集合经常有变化, 那么不但Visitor实现要变化,ConcreteVisitor也要增加相应行为,GOF建议是,不如在这些对象类中直接逐个定义操作,无需使用访问者设计模式。

四、Visitor模式与其它模式

1、如果所浏览的结构对象是线性的,使用迭代模式而不是访问者模式也是可以的
2、访问者模式浏览合成模式的一些结构对象
以上两点来自《Java与模式》一书

五、Visitor模式PHP示例

<?php
 
interface Visitor {
 public function visitConcreteElementA(ConcreteElementA $elementA);
 public function visitConcreteElementB(concreteElementB $elementB);
}
 
interface Element {
 public function accept(Visitor $visitor);
}
 
/**
 * 具体的访问者1
 */
class ConcreteVisitor1 implements Visitor {
 public function visitConcreteElementA(ConcreteElementA $elementA) {
 echo $elementA->getName() . " visitd by ConcerteVisitor1 <br />";
 }
 
 public function visitConcreteElementB(ConcreteElementB $elementB) {
 echo $elementB->getName() . " visited by ConcerteVisitor1 <br />";
 }
 
}
 
/**
 * 具体的访问者2
 */
class ConcreteVisitor2 implements Visitor {
 public function visitConcreteElementA(ConcreteElementA $elementA) {
 echo $elementA->getName() . " visitd by ConcerteVisitor2 <br />";
 }
 
 public function visitConcreteElementB(ConcreteElementB $elementB) {
 echo $elementB->getName() . " visited by ConcerteVisitor2 <br />";
 }
 
}
 
/**
 * 具体元素A
 */
class ConcreteElementA implements Element {
 private $_name;
 
 public function __construct($name) {
 $this->_name = $name;
 }
 
 public function getName() {
 return $this->_name;
 }
 
 /**
 * 接受访问者调用它针对该元素的新方法
 * @param Visitor $visitor
 */
 public function accept(Visitor $visitor) {
 $visitor->visitConcreteElementA($this);
 }
 
}
 
/**
 * 具体元素B
 */
class ConcreteElementB implements Element {
 private $_name;
 
 public function __construct($name) {
 $this->_name = $name;
 }
 
 public function getName() {
 return $this->_name;
 }
 
 /**
 * 接受访问者调用它针对该元素的新方法
 * @param Visitor $visitor
 */
 public function accept(Visitor $visitor) {
 $visitor->visitConcreteElementB($this);
 }
 
}
 
/**
 * 对象结构 即元素的集合
 */
class ObjectStructure {
 private $_collection;
 
 public function __construct() {
 $this->_collection = array();
 }
 
 
 public function attach(Element $element) {
 return array_push($this->_collection, $element);
 }
 
 public function detach(Element $element) {
 $index = array_search($element, $this->_collection);
 if ($index !== FALSE) {
 unset($this->_collection[$index]);
 }
 
 return $index;
 }
 
 public function accept(Visitor $visitor) {
 foreach ($this->_collection as $element) {
 $element->accept($visitor);
 }
 }
}
 
class Client {
 
 /**
 * Main program.
 */
 public static function main() {
 $elementA = new ConcreteElementA("ElementA");
 $elementB = new ConcreteElementB("ElementB");
 $elementA2 = new ConcreteElementB("ElementA2");
 $visitor1 = new ConcreteVisitor1();
 $visitor2 = new ConcreteVisitor2();
 
 $os = new ObjectStructure();
 $os->attach($elementA);
 $os->attach($elementB);
 $os->attach($elementA2);
 $os->detach($elementA);
 $os->accept($visitor1);
 $os->accept($visitor2);
 }
 
}
 
Client::main();
?>

以上就是使用php实现访问者模式的代码,还有一些关于访问者模式的概念区分,希望对大家的学习有所帮助。

PHP 相关文章推荐
PHP学习之正则表达式
Apr 17 PHP
php的memcached客户端memcached
Jun 14 PHP
php操作JSON格式数据的实现代码
Dec 24 PHP
php排序算法(冒泡排序,快速排序)
Oct 09 PHP
基于PHP输出缓存(output_buffering)的深入理解
Jun 13 PHP
深入PHP中的HashTable结构详解
Jun 13 PHP
php实例分享之mysql数据备份
May 19 PHP
PHP中array_slice函数用法实例详解
Nov 25 PHP
使用php转义输出HTML到JavaScript
Mar 27 PHP
PHP文件缓存smarty模板应用实例分析
Feb 26 PHP
thinkPHP模板引擎用法示例
Dec 08 PHP
Laravel下生成验证码的类
Nov 15 PHP
PHP面向对象详解(三)
Dec 07 #PHP
php生成高清缩略图实例详解
Dec 07 #PHP
php冒泡排序与快速排序实例详解
Dec 07 #PHP
PHP常用工具类大全附全部代码下载
Dec 07 #PHP
PHP+ajax分页实例简析
Dec 07 #PHP
php 升级到 5.3+ 后出现的一些错误,如 ereg(); ereg_replace(); 函数报错
Dec 07 #PHP
php+ajax无刷新分页实例详解
Dec 07 #PHP
You might like
PHP图片上传代码
2013/11/04 PHP
PHP页面实现定时跳转的方法
2014/10/31 PHP
最新制作ThinkPHP3.2.3完全开发手册
2015/11/23 PHP
php封装一个异常的处理类
2017/06/08 PHP
PHP实现函数内修改外部变量值的方法示例
2018/12/28 PHP
Laravel项目中timeAgo字段语言转换的改善方法示例
2019/09/16 PHP
jquery实现的鼠标下拉滚动置顶效果
2014/07/24 Javascript
实例解析angularjs的filter过滤器
2016/12/14 Javascript
Web 开发中Ajax的Session 超时处理方法
2017/01/19 Javascript
js推箱子小游戏步骤代码解析
2018/01/10 Javascript
微信小程序实现image组件图片自适应宽度比例显示的方法
2018/01/16 Javascript
解决使用Vue.js显示数据的时,页面闪现原始代码的问题
2018/02/11 Javascript
Koa代理Http请求的示例代码
2018/10/10 Javascript
element-ui带输入建议的input框踩坑(输入建议空白以及会闪出上一次的输入建议问题)
2019/01/15 Javascript
jQuery实现动态生成年月日级联下拉列表示例
2019/05/11 jQuery
简单了解微信小程序的目录结构
2019/07/01 Javascript
详解Nuxt.js中使用Element-UI填坑
2019/09/06 Javascript
js实现弹窗效果
2020/08/09 Javascript
Python3写入文件常用方法实例分析
2015/05/22 Python
python email smtplib模块发送邮件代码实例
2018/04/26 Python
Python+selenium 获取一组元素属性值的实例
2018/06/22 Python
在Django下创建项目以及设置settings.py教程
2019/12/03 Python
Python partial函数原理及用法解析
2019/12/11 Python
使用pyqt5 tablewidget 单元格设置正则表达式
2019/12/13 Python
Tensorflow 实现分批量读取数据
2020/01/04 Python
python中sympy库求常微分方程的用法
2020/04/28 Python
浅谈css3新单位vw、vh、vmin、vmax的使用详解
2017/12/01 HTML / CSS
高级Java程序员面试要点
2013/08/02 面试题
毕业生写求职信的要点
2014/03/04 职场文书
学校光盘行动倡议书
2015/04/28 职场文书
儿童诗两首教学反思
2016/02/23 职场文书
导游词之黄帝陵景区
2019/09/16 职场文书
导游词之湖州-太湖
2019/10/11 职场文书
html5 录制mp3音频支持采样率和比特率设置
2021/07/15 Javascript
Python语言中的数据类型-序列
2022/02/24 Python
mysql使用instr达到in(字符串)的效果
2022/04/03 MySQL