PHP开发学堂:PHP4中实现动态代理
概述
在我们开始实现Dynamic Proxy之前, 也许我们应该先了解一下什么是Proxy和它有什么用. 下面是一篇来自博客堂加菲猫的不错的形象讲述Proxy的文章: “武当学艺与缓存代理”. Proxy模式是”GoF”介绍的23个设计模式之一, Proxy的目的是” Provide a surrogate or placeholder for another object to control access(小型网站之最爱) to it(为其他对象提供一种代理以控制对这个对象的访问)”. 一般常见的代理模式有: 远程代理(Remote Proxy), 虚代理(Virtual Proxy), 保护代理(Protection Proxy), 智能代理(Smart Proxy).
但是使用代理有一个坏处就是你得手动创建所需要代理类的副本(即代理类). 这意味着如果你为Image类创建一个Virtual Proxy, 那么你不得不手动创建一个与Image类有相同Method的ImageProxy类. Ok, 如果你和我一样懒, 你一定会想到动态来产生Proxy. 是的, 接下来你就会发现, 其实在php(做为现在的主流开发语言)4中你很容易实现它.
实现
由于php(做为现在的主流开发语言)4是解释型语言, 弱类型, 且无接口. 所以在实现的时候既有方便之处又有不妥之处. 此处不拘泥于实现方法, 本文也只是实现方法之一.
本文实现的策略其实非常简单. 核心即ProxyFactory类及Clazz类, ProxyFactory负责实例化Clazz, 并赋值. 而由Clazz类负责创建并返回Proxy. 创建Proxy是以写入临时文件方式进行的.
具体请查看ProxyFactory.php(做为现在的主流开发语言)和Clazz.php(做为现在的主流开发语言)二个文件中的代码. 此处不再赘述.
另外在ProxyInvocationHandler.php(做为现在的主流开发语言)中我们定义了一个ProxyInvocationHandler类.
示例
我们现在有一个ReadFileClass类, 该类继承自IReadFileClass, 由于php(做为现在的主流开发语言)4没有接口, 所以此处接口算是模拟的, 事实上在php(做为现在的主流开发语言)4中不使用实现接口也是可行的J. 二个类的具体内容请见清单一和清单二.
清单一
class IReadFileClass
{
function ReadMyFile() {}
}
清单二
class ReadFileClass extends IReadFileClass
{
function ReadMyFile()
{
$fp = fopen(test.txt, "r");
$data = fread($fp, filesize(test.txt));
fclose($fp);
return $data;
}
}
OK, 我们现在要加入验证用户的功能, 即为ReadFileClass中的方法加入保护控制. 如果采用手动创建代理, 那么你可以继承ReadFileClass或者实现IReadFileClass, 并加入保护代码(其实在php(做为现在的主流开发语言)4中甚为自由, 因为除了基本类型外都是object-_-). 不过我们现在试试用刚才实现的动态代理来创建Proxy.
请看清单三的ReadFileClassProxy的代码, 注意该类继承自ProxyInvocationHandler类.
清单三
require_once(ProxyFactory.php(做为现在的主流开发语言));
require_once(ProxyInvocationHandler.php(做为现在的主流开发语言));
require_once(Auth.php(做为现在的主流开发语言));
class ReadFileClassProxy extends ProxyInvocationHandler
{
var $object;
function ReadFileClassProxy(&$obj)
{
$this->object = &$obj;
}
//
function NewInstance(&$obj)
{
$proxyFactory = ProxyFactoryInstance();
return $proxyFactory->create(new ReadFileClassProxy(&$obj),
get_parent_class(&$obj));
}
// $proxy is not used here, but it is useful.
function Invoke(&$proxy, $method, $parameters)
{
$uname = Binzy;
//$uname = Jasmin;
if (Auth::CheckAuth($uname))
{
Return parent::Invoke(&$proxy, $method, $parameters);
}
else
{
//
return No Permission!;
}
}
}
Auth类是一个进行权限验证的类, 此处我们只是简单的查看传入的UserName, 如果是Binzy, 那么自然是可以看秘密的J, 如果是Jasmin, 那么HoHo, 没得看, 给Binzy点空间嘛.:D 详见清单四.
补充:Web开发 , php ,