PHP __call拦截器 实现委托

__call方法可能是最有用的拦截器方法。

当客户端代码要调用类中未定义的方法时,__call会被调用。

__call()接受2个参数,一个是方法的名称,另一个是传递给要调用方法的所有参数(数组)。

__call()方法返回的任何值都会返回给客户,就好像调用一个真实存在的方法一样。

__call()方法对于实现委托也很有用。委托是指一个对象转发或者委托一个请求给另一个对象,被委托的一方替原先对象处理请求。

这类似于继承,和在子类中调用父类的方法有点相似。

但再继承时,父类与子类的关系是固定的,而使用委托则可以再代码运行时改变使用的对象,这意味着委托比继承具有更大的灵活性。

//将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类:

$person = new Person(new PersonWriter());
$person->writeName();