__call方法可能是最有用的拦截器方法。
当客户端代码要调用类中未定义的方法时,__call会被调用。
__call()接受2个参数,一个是方法的名称,另一个是传递给要调用方法的所有参数(数组)。
__call()方法返回的任何值都会返回给客户,就好像调用一个真实存在的方法一样。
__call()方法对于实现委托也很有用。委托是指一个对象转发或者委托一个请求给另一个对象,被委托的一方替原先对象处理请求。
这类似于继承,和在子类中调用父类的方法有点相似。
但再继承时,父类与子类的关系是固定的,而使用委托则可以再代码运行时改变使用的对象,这意味着委托比继承具有更大的灵活性。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 | //将Person类信息格式化并输出 class personWriter{ function writeName(Person $p ){ print $p ->getName(); } function writeAge(Person $p ){ print $p ->getAge(); } } //当然我们可以通过集成PersonWrite类以不同的方式输出Person类的信息。 //下面的的代码结合使用__call()方法和PersonWriter对象来实现Person类: class Person{ private $writer ; function __construct(PersonWriter $write ){ $this ->write = $write ; } function __call( $methodname , $args ){ if (method_exists( $this ->writer, $methodname )){ return $this ->writer-> $methodname ( $this ); } } function getName(){ return "LeoKim" ; }; function getAge(){ return 31; } } |
代码中Person类接受一个PersonWriter对象作为构造方法的参数,并将它存储再属性变量$writer中。
在__call()方法中,我们使用参数$methodname,检查PersonWriter对象中是否存在同名的方法。
如果相应的方法存在,我们就委托PersonWriter对象来处理对方法的调用,把当前类(Person)的实例作为参数传递给PersonWriter对象(使用$this伪变量)。
因此,可以这样调用Person类:
1 2 | $person = new Person( new PersonWriter()); $person ->writeName(); |