直接调用分配给对象属性的闭包

2022-08-30 07:19:14

我希望能够调用我直接分配给对象属性的闭包,而无需将闭包重新分配给变量,然后调用它。这可能吗?

下面的代码不起作用,并导致 .Fatal error: Call to undefined method stdClass::callback()

$obj = new stdClass();
$obj->callback = function() {
    print "HelloWorld!";
};
$obj->callback();

答案 1

从 PHP7 开始,您可以

$obj = new StdClass;
$obj->fn = function($arg) { return "Hello $arg"; };
echo ($obj->fn)('World');

或者使用 Closure::call(),尽管这在 .StdClass


在 PHP7 之前,您必须实现 magic 方法来拦截调用并调用回调(这当然是不可能的,因为您无法添加该方法)__callStdClass__call

class Foo
{
    public function __call($method, $args)
    {
        if(is_callable(array($this, $method))) {
            return call_user_func_array($this->$method, $args);
        }
        // else throw exception
    }
}

$foo = new Foo;
$foo->cb = function($who) { return "Hello $who"; };
echo $foo->cb('World');

请注意,您不能这样做

return call_user_func_array(array($this, $method), $args);

在体内,因为这会在无限循环中触发。__call__call


答案 2

您可以通过在闭包上调用__invoke来执行此操作,因为这是对象用来像函数一样运行的神奇方法:

$obj = new stdClass();
$obj->callback = function() {
    print "HelloWorld!";
};
$obj->callback->__invoke();

当然,如果回调是数组或字符串(在PHP中也可以是有效的回调),这将不起作用 - 仅适用于闭包和其他具有__invoke行为的对象。


推荐