PHP设计模式-对象行为型模式-VISITOR

发表于:2007-07-04来源:作者:点击数: 标签:
个人认为在23个经典模式中VISITOR是比较难理解的一个,所以决定先讲讲自己对VISITOR的理解。因为马哲认为对事物的认识是从具体到抽象的一个过程,所以在谈理论之前先说一个例子是必要的。这个例子不是实际应用,但我想它还算生动,是个记忆VISITOR模式的好例

    个人认为在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不能重载,哭一下]

原文转自:http://www.ltesting.net