因为某个对象消耗太多资源,而且你的代码并不是每个逻辑路径都需要此对象, 你曾有过延迟创建对象的想法吗 ( if和else就是不同的两条逻辑路径) ? 你有想过限制访问某个对象,也就是说,提供一组方法给普通用户,特别方法给管理员用户?以上两种需求都非常类似,并且都需要解决一个更大的问题:你如何提供一致的接口给某个对象让它可以改变其内部功能,或者是从来不存在的功能?
问题:
你怎样才能在不直接操作对象的情况下,对此对象进行访问?
解决方案
代理模式是给某一个对象提供一个替代者(占位者),使之在client对象和subject对象之间编码更有效率。代理可以提供延迟实例化(lazy instantiation),控制访问, 等等,包括只在调用中传递。 一个处理纯本地资源的代理有时被称作虚拟代理。远程服务的代理常常称为远程代理。强制控制访问的代理称为保护代理。
下面有一个关于远程代理的图(只有一个方法)。 SoapClient是本地对象(客户端)的媒介,通过调用SoapServer(subject)来获得天气信息。全部任务都通过HTTP协议达成,比如创建,传输,接收,通过SoapClient类内部的功能解析复杂XML文档完成远程通信。网络返回的结果跟直接调用SoapServer对象API的效果一样的,因此SoapClient可视为一个本地的代替者(代理),来获得远程SoapServer的资源。
这里有其他不同种类的代理模式叫做smart 代理。它的作用是:在允许访问subject对象之前增加一些各种各样的附加逻辑(additional logic)。(译注:这里的附加逻辑是指在建立了代理模式的基本结构之后,根据自己的需求在代理模式中添加的代码)
注:Handle-Body 模式
代理模式, 装饰器模式, 和适配器模式从编码角度看,都有类似的结构的(后两个模式在下面两章讲)。
三种模式本质区别在于如何使用它们。
这个结构的其他变种在下面网址可以找到:http://www。c2。com/cgi/wiki?HandleBodyPattern。
理的本质是用一个实例化变量对subject对象进行引用,通过调用代理类的方法操作subject类。
让我们看一个代理模式的最简单形态,首先,你需要创建一个subject类用于代理。
| // PHP4 class Subject { function someMethod() { sleep(1); //do something } } |
下一步,你需要一个代理类,这个类需要实例化subject类用于代理。
| class ProxySubject { var $subject; function ProxySubject() { $this->subject =& new Subject; } } |
在上面的ProxySubject类,在构造器中创建了subject对象(还有其他的可供选择的方法,比如通过传递一个参数给构造器或者用工厂创建subject对象,这些都是同样可行的)。
最后,你的代理类提供所有公有的方法必须让subject类支持。在这个案例上, someMethod()就是这样一个方法。
| class ProxySubject { var $subject; function ProxySubject() { $this->subject =& new Subject; } function someMethod() { $this->subject->someMethod(); } } |
ProxySubject类通过$this->subject->someMethod()才真正的调用Subject类。
代理既可以有一些方法直接调用,又可以在调用之前使用一些附加逻辑(延迟加载,监视)。
这里用了一个UML类图表示ProxySubject类:
一个简单的例子
上面的简单例子展示了代理模式的基本结构,当然我们需要一些更有趣和实际的例子。
Web服务变得非常流行,PHP5包含了一些支持的很好的协议,就如SOAP一样可以很容易的理解远程服务。 创建SOAP客户端的部分功能是为了处理WSDL文件。然而,你可以延迟处理WSDL文件直到你需要处理这个文件的时候。席面一个代理的例子将会展示远程代理访问SOAP服务和延迟实例化。
远程代理
首先,基于PHP5风格, 来一段创建简单的SoapClient对象的代码。你必须编译的时候加上—enable-soap选项,才能使用SoapClient类,如果你已经做过了,那么你就可以用URL形式,把WSDL文件传入构造器来创建SoapClient实例:
| // PHP5 $client = new SoapClient( ‘http://live。capescience。com/wsdl/GlobalWeather。wsdl’); |
注:PHP4风格的SoapClients
在你编码PHP4风格的SOAP客户端之前,PHP5的技术可以忽略。PHP5 的SoapClient是一个扩展,所以它是原生的PHP的代码,速度更加快(译注:这里的原生PHP代码应该是原生代码,原生代码是指编译性语言编写的代码),实际上是用C语言完成解析和格式化XML信息的功能。
PHP4风格 SOAP库包括:
| ? phpsoaptoolkit (http://phpsoaptoolkit。sf。net/phpsoap/), ? PEAR::SOAP (http://pear。php。net/package/SOAP) ? ez SOAP (http://ez。no/ez_publish/documentation/development/libraries/ez_soap) ? nusoap (http://sf。net/projects/nusoap/)。 |
所有这些php4的库在处理远程信息的格式化和传递的功能是使用PHP代码实现的,并且有远程
代理的例子。
首先一个问题是你用什么方法让SoapClient做回应? 运行var_dump(get_class_methods(get_class($client)));,你可以很容易的列举在运行时的方法。需要更加详细的例子的话,你可以参考下面的测试案例:
| class ProxyTestCase extends UnitTestCase { const WSDL = ‘http://live。capescience。com/wsdl/GlobalWeather。wsdl’; private $client; function setUp() { $this->client = new SoapClient(ProxyTestCase::WSDL); } function TestMethodsOfSoapClient() { $soap_client_methods = array( ‘__construct’, ‘__call’, ‘__soapCall’, ‘__getLastRequest’, ‘__getLastResponse’, ‘__getLastRequestHeaders’, ‘__getLastResponseHeaders’, ‘__getFunctions’, ‘__getTypes’, ‘__doRequest’); $this->assertEqual( $soap_client_methods, get_class_methods(get_class($this->client))); } } |
咋一看,似乎写了一个没有用的测试,难道你只是为了在任意时候显示这些信息而已?或许吧,在PHP升级的时候,这个测试放入程序进行测试对于监视你的程序会很有用,比如发现有什么方法增加了,或者是你可以发现哪些被依赖的方法被删除了,验证PHP编译的时候是否加入了SOAP选项。但必须要说的是,这个测试是极端的脆弱:其弱点就是会因为更改代码的原因,需要重构而且高度依赖函数列表的顺序。目前,虽然这个测
















