Как перехватить вызов любого метода объекта в PHP?
-
15-09-2020 - |
Вопрос
Я пытаюсь понять, как перехватить любой метод, вызываемый для объекта в 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
Есть ли способ, как это сделать?Помоги пожалуйста :)
Решение
Берем свой оригинал 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();
Другие советы
Вы можете обернуть объект вокруг объекта, перехватывая любые вызовы, затем пересылают их на исходный объект и возврат результата.
Просто храните объект в качестве переменной в классе Wrapper и используйте методы перегрузки в своем классе обертки, чтобы позвонить / установить / получить / проверять на объекте.
$object = new AnyObject;
$object = new Wrapper($object);
$object->anyMethod();
$object->anyVar = 'test';
echo $object->anyVar;
echo $object['array form'];
.
Завораживание класса обертки в Foreach, вероятно, сложнее.Не пробовал это.
Если вы устанавливаете функцию в Private, вызов перехватите любой звонок на него снаружи, будет в ловушке __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
}
}
.