个人认为在23个经典模式中VISITOR是比较难理解的一个,所以决定先讲讲自己对VISITOR的理解。因为马哲认为对事物的认识是从具体到抽象的一个过程,所以在谈理论之前先说一个例子是必要的。这个例子不是实际应用,但我想它还算生动,是个记忆VISITOR模式的好例子吧。
英国、美国都有自己的核武机构,每个国家的核武机构,都使用不同的接口来进行通讯:
CODE:
class Country
{
// ...
}
1,对于英国,核武密码被分成三个部分,需要用三个接口取得:
CODE:
interface I1{function get1();};
interface I2{function get2();};
interface I3{function get3();};
class English extends Country implements I1,I2,I3
{
function get1(){return '123';}
function get2(){return '456';}
function get3(){return '789';}
}
2,对于美国,保管核武密码的方式不一样,使用了5个接口来分别取得密码的部分:
CODE:
interface Ia{function getA();};
interface Ib{function getB();};
interface Ic{function getC();};
interface Id{function getD();};
interface Ie{function getE();};
class America extends Country implements Ia,Ib,Ic,Id,Ie
{
function getA(){return 'a';}
function getB(){return 'b';}
function getC(){return 'c';}
function getD(){return 'd';}
function getE(){return 'e';}
}
如果我们中国欲取得此二国的核武密码,则必须先熟知这二国的核武密码接口。实际行动(runtime)时,
用if..else来判断现在具体是哪个国家,然后调用该国相应的核武密码接口。
CODE:
class Client
{
private function getRealPwd(Country $country)
{
if($country instanceof English)
{
return $country->get1() mod ( $country->get2() + $country->get3() );
}elseif($country instanceof American){
return $country->getA() . $country->getB() . $country->getC() . $country->getD() . $country->getE();
}elseif(...){
...
}
}
}
这样写是不错,但是有个缺点,就是每次都要用IF..ELSE进行判断,各种不同的接口,各种不同的密码组合算法,国家一旦多起来,某些领导(Client)就嫌太麻烦了。
于是,某些聪明的领导便在分别英、美花钱贿络,新开了一个方便的接口Aclearcase/" target="_blank" >ccept():
CODE:
interface IAccept{function Accept(Visitor $Visitor);}
英国:
class English extends Country implements IEnglishPwd1,IEnglishPwd1,IEnglishPwd1,IAccept
{
....
function Accept(Visitor $Visitor)
{
return $Visitor->VisitorObject($this);
}
}
美国:
class America extends Country implements a,b,c,d,e,IAccept
{
....
function Accept(Visitor $Visitor)
{
return $Visitor->VisitorObject($this);
}
}
然后,聪明的领导,培训了一个密码特务(PasswordVisitor),可以针对不同的国家,使用不同的组合密码算法:
CODE:
class PasswordVisitor
{
function VisitorObject(English $Country)
{
return $Country->get1() mod ( $Country->get2() + $Country->get3() );
}
function VisitorObject(America $Country)
{
return $Country->getA() . $Country->getB() . $Country->getC() . $Country->getD() . $Country->getE();
}
}
这下好了,领导不须理会具体是哪个国家了,只须打开该国的Accept()接口,然后把密码特务扔进去,就能得到核武密码了:
CODE:
class Client
{
private function getRealPwd(Country $country)
{
return $country->Accept(new PasswordVisitor());
}
}
这样,聪明的领导就悠闲了,可以喝咖啡看报纸去了。
当然,敌国的情报,不只有核武密码,于是,就出现了许多种类型的间谍Visitor,我们从中抽象出这个接口:
CODE:
public interface IVisitor
{
string VisitorObject(Country $c);
string VisitorObject(Country $c);
}
然后,通过扩展IVistor接口,定义不同类型的间谍,然后放进Access()这个方便的通道中,就得到不同的返回资料,这就好象是在对国家定义一系列新的操作,
这就是原文所说:“它可以使你在不改变这些对象本身的情况下,定义作用于这些对象的新操作”
[*注:其实那个Vistor是错的,因为PHP不能重载,哭一下]