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 相关文章推荐
PHPMailer使用教程(PHPMailer发送邮件实例分析)
Dec 06 PHP
php中call_user_func函数使用注意事项
Nov 21 PHP
php文件上传简单实现方法
Jan 24 PHP
PHP滚动日志的代码实现
Jun 10 PHP
PHP中substr函数字符串截取用法分析
Jan 07 PHP
Yii调试查看执行SQL语句的方法
Jul 15 PHP
PHP中函数gzuncompress无法使用的解决方法
Mar 02 PHP
PHP简单获取上月、本月、近15天、近30天的方法示例
Jul 03 PHP
thinkphp3.2嵌入百度编辑器ueditor的实例代码
Jul 13 PHP
Yii2框架中使用PHPExcel导出Excel文件的示例
Aug 09 PHP
PHP实现基于3DES算法加密解密字符串示例
Aug 24 PHP
php 自定义函数实现将数据 以excel 表格形式导出示例
Nov 13 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中使用$_REQUEST需要注意的一个问题
2013/05/02 PHP
php的ajax简单实例
2014/02/27 PHP
PHP fopen()和 file_get_contents()应用与差异介绍
2014/03/19 PHP
在SAE上搭建最新wordpress的方法
2014/12/21 PHP
php使用PDO从数据库表中读取数据的实现方法(必看)
2017/06/02 PHP
PHP更安全的密码加密机制Bcrypt详解
2017/06/18 PHP
Aster vs KG BO3 第一场2.19
2021/03/10 DOTA
JQuery防止退格键网页后退的实现代码
2012/03/23 Javascript
Extjs407 getValue()和getRawValue()区别介绍
2013/05/21 Javascript
js与jquery获取父元素,删除子元素的两种不同方法
2014/01/09 Javascript
jquery禁止输入数字以外的字符的示例(纯数字验证码)
2014/04/10 Javascript
javascript获取flash版本号的方法
2014/11/20 Javascript
JavaScript操作URL的相关内容集锦
2015/10/29 Javascript
任意Json转成无序列表的方法示例
2016/12/09 Javascript
bootstrap PrintThis打印插件使用详解
2017/02/20 Javascript
three.js绘制地球、飞机与轨迹的效果示例
2017/02/28 Javascript
Node.js Mongodb 密码特殊字符 @的解决方法
2017/04/11 Javascript
详解vue-cli之webpack3构建全面提速优化
2017/12/25 Javascript
python中异常报错处理方法汇总
2016/11/20 Python
python爬取Ajax动态加载网页过程解析
2019/09/05 Python
详解Python3 pickle模块用法
2019/09/16 Python
python之生成多层json结构的实现
2020/02/27 Python
基于Python编写一个计算器程序,实现简单的加减乘除和取余二元运算
2020/08/05 Python
解决pycharm修改代码后第一次运行不生效的问题
2021/02/06 Python
Sneaker Studio波兰:购买运动鞋
2018/04/28 全球购物
Hotels.com拉丁美洲:从豪华酒店到经济型酒店的预定优惠和折扣
2019/12/09 全球购物
htmlentities() 和 htmlspecialchars()有什么区别
2015/07/01 面试题
介绍一下linux的文件系统
2012/03/20 面试题
Linux上比较文件的命令都有哪些
2012/02/24 面试题
致铅球运动员加油稿
2014/02/13 职场文书
临床医师个人自我评价
2014/04/06 职场文书
2015年感恩母亲节活动方案
2015/05/04 职场文书
2015年大学迎新晚会总结
2015/07/16 职场文书
LayUI+Shiro实现动态菜单并记住菜单收展的示例
2021/05/06 Javascript
OpenCV-Python模板匹配人眼的实例
2021/06/08 Python
Django路由层如何获取正确的url
2021/07/15 Python