php设计模式之委托模式
程序员文章站
2024-04-03 17:34:28
委托模式是软件设计模式中的一项基本技巧。在委托模式中,有两个对象参与处理同一个请求,接受请求的对象将请求委托给另一个对象来处理。委托模式是一项基本技巧,许多其他的模式,如状...
委托模式是软件设计模式中的一项基本技巧。在委托模式中,有两个对象参与处理同一个请求,接受请求的对象将请求委托给另一个对象来处理。委托模式是一项基本技巧,许多其他的模式,如状态模式、策略模式、访问者模式本质上是在更特殊的场合采用了委托模式。
动态委托的介绍:动态委托概念来自于jakarta 字节码工程库 (byte-code engineering library, bcel)。它能够分析存在的类,并且对于接口,抽象类,甚至运行时的具体类来说,它能够生成以字节编码委托类。
被委托的接口/类应该满足如下条件:动态委托最多只能委托一个类,但是能够代理多个接口。这个限制来自于java的单继承模式。一个java类最多只有一个父类。既然生成的委托类把被委托类作为它的父类,那么指定多个被委托类是不合理的。如果没有指定被委托类,那么缺省的父类就是object。
下面是php 反射机制实现动态代理的代码:
<?php class fruit { function callfruit() { print "generate an apple"; } } class fruitdelegator { private $targets; function __construct() { $this->target[] = new fruit(); } function __call($name, $args) { foreach ($this->target as $obj) { $r = new reflectionclass($obj); if ($method = $r->getmethod($name)) { if ($method->ispublic() && !$method->isabstract()) { return $method->invoke($obj, $args); } } } } } $obj = new fruitdelegator(); $obj->callfruit(); // 运行结果 // generate an apple ?>
可见,通过代理类fruitdelegator来代替fruit类来实现他的方法。
同样的,如下的代码也是能够运行的:
<?php class color { function callcolor() { print "generate red"; } } class colordelegator { private $targets; function addobject($obj) { $this->target[] = $obj; } function __call($name, $args) { foreach ($this->target as $obj) { $r = new reflectionclass($obj); if ($method = $r->getmethod($name)) { if ($method->ispublic() && !$method->isabstract()) { return $method->invoke($obj, $args); } } } } } $obj = new colordelegator(); $obj->addobject(new color()); $obj->callcolor(); ?>
设计了一个cd类,类中有mp3播放模式,和mp4播放模式
改进前,使用cd类的播放模式,需要在实例化的类中去判断选择什么方式的播放模式
改进后,播放模式当做一个参数传入playlist函数中,就自动能找到对应需要播放的方法。
一、未改进前
<?php //使用委托模式之前,调用cd类,选择cd播放模式是复杂的选择过程 class cd { protected $cdinfo = array(); public function addsong($song) { $this->cdinfo[$song] = $song; } public function playmp3($song) { return $this->cdinfo[$song] . '.mp3'; } public function playmp4($song) { return $this->cdinfo[$song] . '.mp4'; } } $oldcd = new cd; $oldcd->addsong("1"); $oldcd->addsong("2"); $oldcd->addsong("3"); $type = 'mp3'; if ($type == 'mp3') { $oldcd->playmp3(); } else { $oldcd->playmp4(); }
二、通过委托模式,改进后的cd类
<?php namespace tools; /* 委托模式 去除核心对象中的判决和复杂功能性 */ //委托接口 interface delegate{ public function playlist($list,$song); } //mp3处理类 class mp3 implements delegate{ public function playlist($list,$song){ return $list[$song].'.mp3'; } } //mp4处理类 class mp4 implements delegate{ public function playlist($list, $song) { return $list[$song].'.mp4'; } } class cddelegate{ protected $cdinfo = array(); public function addsong($song){ $this->cdinfo[$song] = $song; } public function play($type,$song){ $name = '\tools\\'.$type; $obj = new $name; return $obj->playlist($this->cdinfo,$song); } } $newcd = new cddelegate(); $newcd->addsong("1"); $newcd->addsong("2"); $newcd->addsong("3"); echo $newcd->play('mp3','1');//只要传递参数就能知道需要选择何种播放模式
再为大家分享一个实例:
<?php /** * 委托模式 示例 * * @create_date: 2010-01-04 */ class playlist { var $_songs = array(); var $_object = null; function playlist($type) { $object = $type."playlistdelegation"; $this->_object = new $object(); } function addsong($location,$title) { $this->_songs[] = array("location"=>$location,"title"=>$title); } function getplaylist() { return $this->_object->getplaylist($this->_songs); } } class mp3playlistdelegation { function getplaylist($songs) { $aresult = array(); foreach($songs as $key=>$item) { $path = pathinfo($item['location']); if(strtolower($item['extension']) == "mp3") { $aresult[] = $item; } } return $aresult; } } class rmvbplaylistdelegation { function getplaylist($songs) { $aresult = array(); foreach($songs as $key=>$item) { $path = pathinfo($item['location']); if(strtolower($item['extension']) == "rmvb") { $aresult[] = $item; } } return $aresult; } } $omp3playlist = new playlist("mp3"); $omp3playlist->getplaylist(); $ormvbplaylist = new playlist("rmvb"); $ormvbplaylist->getplaylist(); ?>
以上就是本文的全部内容,希望对大家的学习有所帮助。