委托模式是軟件設計模式中的一項基本技巧。在委托模式中,有兩個對象參與處理同一個請求,接受請求的對象將請求委托給另一個對象來處理。委托模式是一項基本技巧,許多其他的模式,如狀態模式、策略模式、訪問者模式本質上是在更特殊的場合采用了委托模式。
動態委托的介紹:動態委托概念來自於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();
?>