Como pegar qualquer chamada de método em um objeto em PHP?
-
15-09-2020 - |
Pergunta
Eu estou tentando descobrir como pegar qualquer método chamado em um objeto em PHP.Eu sei sobre a magia função __call
, mas ele é disparado somente para métodos que não existem no chamado objeto.
Por exemplo, eu tenho algo parecido com isto:
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
}
}
E quando eu faço isso:
$foo = new Foo();
$foo->bar();
Eu quero essa saída:
Calling method bar
foobar
em vez de um presente:
foobar
Existe alguma maneira de como fazer isso?Ajuda por favor :)
Solução
Tendo original Foo
implementação você pode moldar um decorator
em torno dele, como este:
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();
Outras dicas
Você pode moldar um objeto em torno do objeto de interceptação de chamadas, em seguida, encaminhá-los sobre o objeto original e devolver o resultado.
Basta armazenar o objeto como uma variável na classe de wrapper e use sobrecarga de métodos na sua classe de wrapper para chamar/set/get/check-no objeto.
$object = new AnyObject;
$object = new Wrapper($object);
$object->anyMethod();
$object->anyVar = 'test';
echo $object->anyVar;
echo $object['array form'];
Ciclo a classe de wrapper em foreach é provavelmente mais difícil.Ainda não tentei isso.
Se você definir a função privada , chamada de interceptação de qualquer chamada do lado de fora vai ser preso no __chamar, mas você pode chamá-lo a partir do interior
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
}
}