PHP设计模式之责任链模式的深入解析


Posted in PHP onJune 13, 2013

责任链模式,其目的是组织一个对象链处理一个如方法调用的请求。
当ConcreteHandler(具体的处理程序)不知道如何满足来自Client的请求时,或它的目的不是这个时,它会委派给链中的下一个Handler(处理程序)来处理。

这个设计模式通常和复合模式一起使用,其中有些叶子或容器对象默认委派操作给它们的父对象。另一个例子是,本地化通常是使用责任链处理的,当德语翻译适配器没有为翻译关键词找到合适的结果时,就返回到英语适配器或干脆直接显示关键词本身。

耦合减少到最低限度:Client类不知道由哪个具体的类来处理请求;在创建对象图时配置了链;ConcreteHandlers不知道哪个对象是它们的继承者。行为在对象之间分配是成功的,链中最近的对象有优先权和责任满足请求。

参与者:
◆Client(客户端):向Handler(处理程序)提交一个请求;
◆Handler(处理程序)抽象:接收一个请求,以某种方式满足它;
◆ConcreteHandlers(具体的处理程序):接收一个请求,设法满足它,如果不成功就委派给下一个处理程序。
下面的代码实现了一个最著名的责任链示例:多级缓存。

/**  
 * The Handler abstraction. Objects that want to be a part of the  
 * ChainOfResponsibility must implement this interface directly or via  
 * inheritance from an AbstractHandler.  
 */ 
interface KeyValueStore  
{  
    /**  
     * Obtain a value.  
     * @param string $key  
     * @return mixed  
     */ 
    public function get($key);  
}  /**  
 * Basic no-op implementation which ConcreteHandlers not interested in  
 * caching or in interfering with the retrieval inherit from.  
 */ 
abstract class AbstractKeyValueStore implements KeyValueStore  
{  
    protected $_nextHandler;  
    public function get($key)  
    {  
 return $this->_nextHandler->get($key);  
    }  
}  
/**  
 * Ideally the last ConcreteHandler in the chain. At least, if inserted in  
 * a Chain it will be the last node to be called.  
 */ 
class SlowStore implements KeyValueStore  
{  
    /**  
     * This could be a somewhat slow store: a database or a flat file.  
     */ 
    protected $_values;  
    public function __construct(array $values = array())  
    {  
 $this->_values = $values;  
    }  
    public function get($key)  
    {  
 return $this->_values[$key];  
    }  
}  
/**  
 * A ConcreteHandler that handles the request for a key by looking for it in  
 * its own cache. Forwards to the next handler in case of cache miss.  
 */ 
class InMemoryKeyValueStore implements KeyValueStore  
{  
    protected $_nextHandler;  
    protected $_cached = array();  
    public function __construct(KeyValueStore $nextHandler)  
    {  
 $this->_nextHandler = $nextHandler;  
    }  
    protected function _load($key)  
    {  
 if (!isset($this->_cached[$key])) {  
     $this->_cached[$key] = $this->_nextHandler->get($key);  
 }  
    }  
    public function get($key)  
    {  
 $this->_load($key);  
 return $this->_cached[$key];  
    }  
}  
/**  
 * A ConcreteHandler that delegates the request without trying to  
 * understand it at all. It may be easier to use in the user interface  
 * because it can specialize itself by defining methods that generates  
 * html, or by addressing similar user interface concerns.  
 * Some Clients see this object only as an instance of KeyValueStore  
 * and do not care how it satisfy their requests, while other ones  
 * may use it in its entirety (similar to a class-based adapter).  
 * No client knows that a chain of Handlers exists.  
 */ 
class FrontEnd extends AbstractKeyValueStore  
{  
    public function __construct(KeyValueStore $nextHandler)  
    {  
 $this->_nextHandler = $nextHandler;  
    }  
    public function getEscaped($key)  
    {  
 return htmlentities($this->get($key), ENT_NOQUOTES, 'UTF-8');  
    }  
}  
// Client code  
$store = new SlowStore(array('pd' => 'Philip K. Dick',  
 'ia' => 'Isaac Asimov',  
 'ac' => 'Arthur C. Clarke',  
 'hh' => 'Helmut Heißenbüttel'));  
// in development, we skip cache and pass $store directly to FrontEnd  
$cache = new InMemoryKeyValueStore($store);  
$frontEnd = new FrontEnd($cache);  
echo $frontEnd->get('ia'), "\n";  
echo $frontEnd->getEscaped('hh'), "\n";

关于PHP责任链设计模式的一些实现说明:
◆责任链可能已经存在于对象图中,和复合模式的例子一样;
◆此外,Handler抽象可能存在,也可能不存在,最好的选择是一个分开的Handler接口只可以执行handleRequest()操作,不要强制一个链只在一个层次中,因为后面的已经存在了;
◆也可能引入一个抽象类,但由于请求处理是一个正交关注,因此具体的类可能已经继承了其它类;
◆通过constructor 或setter,Handler(或下一个Handler)被注入到Client或前一个Handler;
◆请求对象通常是一个ValueObject,也可能被实现为一个Flyweight,在PHP中,它可能是一个标量类型,如string,注意在某些语言中,一个string就是一个不变的ValueObject。

简单的总结责任链模式,可以归纳为:用一系列类(classes)试图处理一个请求request,这些类之间是一个松散的耦合,唯一共同点是在他们之间传递request. 也就是说,来了一个请求,A类先处理,如果没有处理,就传递到B类处理,如果没有处理,就传递到C类处理,就这样象一个链条(chain)一样传递下去。

PHP 相关文章推荐
用libTemplate实现静态网页的生成
Oct 09 PHP
PHP的单引号和双引号 字符串效率
May 27 PHP
PHP 的ArrayAccess接口 像数组一样来访问你的PHP对象
Oct 12 PHP
php 变量未定义等错误的解决方法
Jan 12 PHP
PHP FATAL ERROR: CALL TO UNDEFINED FUNCTION BCMUL()解决办法
May 04 PHP
php实现的发送带附件邮件类实例
Sep 22 PHP
隐性调用php程序的方法
Jun 13 PHP
php实现上传图片文件代码
Jul 19 PHP
非常全面的php日期时间运算汇总
Nov 04 PHP
PHP的Yii框架中使用数据库的配置和SQL操作实例教程
Mar 17 PHP
php版微信js-sdk支付接口类用法示例
Oct 12 PHP
CI框架实现框架前后端分离的方法详解
Dec 30 PHP
PHP设计模式之结构模式的深入解析
Jun 13 #PHP
PHP设计模式之命令模式的深入解析
Jun 13 #PHP
深入Memcache的Session数据的多服务器共享详解
Jun 13 #PHP
探讨:如何使用PHP实现计算两个日期间隔的年、月、周、日数
Jun 13 #PHP
判断php数组是否为索引数组的实现方法
Jun 13 #PHP
深入解析yii权限分级式访问控制的实现(非RBAC法)
Jun 13 #PHP
PHP 基于Yii框架中使用smarty模板的方法详解
Jun 13 #PHP
You might like
利用PHP实现智能文件类型检测的实现代码
2011/08/02 PHP
Zend的MVC机制使用分析(二)
2013/05/02 PHP
PHP实现的折半查找算法示例
2017/12/19 PHP
基于jquery的jqDnR拖拽溢出的修改
2011/02/12 Javascript
JavaScript基础篇之变量作用域、传值、传址的简单介绍与实例
2013/06/29 Javascript
JS将表单导出成EXCEL的实例代码
2013/11/11 Javascript
jQuery中slideUp()方法用法分析
2014/12/24 Javascript
jquery 插件实现瀑布流图片展示实例
2015/04/03 Javascript
浅析JS异步加载进度条
2016/05/05 Javascript
微信jssdk用法汇总
2016/07/16 Javascript
node.js与C语言 实现遍历文件夹下最大的文件,并输出路径,大小
2017/01/20 Javascript
Bootstrap媒体对象学习使用
2017/03/07 Javascript
javascript实现的图片预览功能
2017/03/25 Javascript
javascript 中的try catch应用总结
2017/04/01 Javascript
nodejs实现连接mongodb数据库的方法示例
2018/03/15 NodeJs
详解如何在微信小程序开发中正确的使用vant ui组件
2018/09/13 Javascript
解决Vue在封装了Axios后手动刷新页面拦截器无效的问题
2018/11/08 Javascript
laydate时间日历插件使用方法详解
2018/11/14 Javascript
微信小程序 select 下拉框组件功能
2019/09/09 Javascript
Python cv2 图像自适应灰度直方图均衡化处理方法
2018/12/07 Python
分享Python切分字符串的一个不错方法
2018/12/14 Python
简单了解Python生成器是什么
2019/07/02 Python
数控技术与应用毕业生自荐信
2013/09/24 职场文书
销售业务员岗位职责
2014/01/29 职场文书
《蚂蚁和蝈蝈》教学反思
2014/02/24 职场文书
自考毕业自我鉴定
2014/03/18 职场文书
党支部公开承诺书
2014/03/28 职场文书
党员三严三实对照检查材料
2014/10/13 职场文书
社区务虚会发言材料
2014/10/20 职场文书
观后感开头
2015/06/19 职场文书
学生病假条怎么写
2015/08/17 职场文书
小学班主任工作经验交流材料
2015/11/02 职场文书
领导激励员工的演讲稿,各种会上用得到,建议收藏
2019/08/13 职场文书
Redis Cluster集群动态扩容的实现
2021/07/15 Redis
Go 中的空白标识符下划线
2022/03/25 Golang
Golang解析JSON对象
2022/04/30 Golang