如何在PHP中捕获对象的任何方法调用?

时间:2010-07-13 22:08:43

标签: php object interception

我试图找出如何捕获在PHP中调用对象的任何方法。我知道魔术函数__call,但它仅针对被调用对象上不存在的方法触发。

例如我有这样的事情:

class Foo
{
  public function bar()
  {
    echo 'foobar';
  }

  public function override($method_name,$method_args)
  {
    echo 'Calling method ',$method_name,'<br />';
    $this->$method_name($method_args); //dirty, but working
  }
}

当我这样做时:

$foo = new Foo();
$foo->bar();

我想要这个输出:

Calling method bar
foobar

而不是这一个:

foobar

有什么方法可以做到这一点吗?请帮助:)

3 个答案:

答案 0 :(得分:45)

采用您原来的Foo实现,您可以将decorator包裹起来,如下所示:

class Foo 
{
    public function bar() {
        echo 'foobar';
    }
}

class Decorator 
{
    protected $foo;

    public function __construct(Foo $foo) {
       $this->foo = $foo;
    }

    public function __call($method_name, $args) {
       echo 'Calling method ',$method_name,'<br />';
       return call_user_func_array(array($this->foo, $method_name), $args);
    }
}

$foo = new Decorator(new Foo());
$foo->bar();

答案 1 :(得分:2)

您可以在对象周围包装对象,拦截任何调用,然后在原始对象上转发它们并返回结果。

只需将对象存储为包装类中的变量,并在包装​​类中使用重载方法来调用/设置/获取/检查对象。

$object = new AnyObject;
$object = new Wrapper($object);

$object->anyMethod();
$object->anyVar = 'test';
echo $object->anyVar;
echo $object['array form'];

在foreach中循环包装类可能更难。没试过。

答案 2 :(得分:0)

如果你把这个功能设置为私有,那么调用会从外面捕获任何对它的调用都会被困在__call中,但你可以从里面调用它

class Foo
{
   private function bar()
   {
      echo 'foobar';
   }

   public function __call($method_name,$method_args)
   {
      echo 'Calling method ',$method_name,'<br />';
      $this->$method_name(); //dirty, but working
   }
}