클로저를 변수에 다시 할당 한 다음 호출하지 않고 객체의 속성에 직접 할당 한 클로저를 호출 할 수 있기를 원합니다. 이게 가능해?
아래 코드는 작동하지 않으며 Fatal error: Call to undefined method stdClass::callback()
.
$obj = new stdClass();
$obj->callback = function() {
print "HelloWorld!";
};
$obj->callback();
답변
PHP7부터 다음을 수행 할 수 있습니다.
$obj = new StdClass;
$obj->fn = function($arg) { return "Hello $arg"; };
echo ($obj->fn)('World');
또는 Closure :: call () 을 사용하지만 StdClass
.
PHP7 이전 __call
에는 호출을 가로 채서 콜백을 호출 하는 매직 메서드 를 구현해야했습니다 ( StdClass
물론 __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
무한 루프.
답변
클로저에서 __invoke를 호출하여이를 수행 할 수 있습니다. 이는 객체가 함수처럼 동작하는 데 사용하는 마법 메서드이기 때문입니다.
$obj = new stdClass();
$obj->callback = function() {
print "HelloWorld!";
};
$obj->callback->__invoke();
물론 콜백이 배열이나 문자열 (PHP에서도 유효한 콜백 일 수 있음) 인 경우에는 작동하지 않습니다. 클로저 및 __invoke 동작이있는 다른 객체에만 해당됩니다.
답변
현재 PHP 7 다음을 수행 할 수 있습니다 :
($obj->callback)();
답변
PHP 7 부터는 다음 call()
메소드를 사용하여 클로저를 호출 할 수 있습니다 .
$obj->callback->call($obj);
PHP 7 은 임의의 (...)
표현식에 대해서도 연산을 실행할 수 있기 때문에 ( Korikulum에서 설명한 대로 ) :
($obj->callback)();
다른 일반적인 PHP 5 접근 방식은 다음과 같습니다.
-
매직 방법 사용
__invoke()
( Brilliand 에서 설명 )$obj->callback->__invoke();
-
은 USING
call_user_func()
함수call_user_func($obj->callback);
-
표현식에서 중간 변수 사용
($_ = $obj->callback) && $_();
각 방식마다 장단점이 있지만 가장 급진적이고 확실한 솔루션은 여전히 Gordon이 제시 한 솔루션 입니다.
class stdKlass
{
public function __call($method, $arguments)
{
// is_callable([$this, $method])
// returns always true when __call() is defined.
// is_callable($this->$method)
// triggers a "PHP Notice: Undefined property" in case of missing property.
if (isset($this->$method) && is_callable($this->$method)) {
return call_user_func($this->$method, ...$arguments);
}
// throw exception
}
}
$obj = new stdKlass();
$obj->callback = function() { print "HelloWorld!"; };
$obj->callback();
답변
을 사용하여 가능한 것 같습니다 call_user_func()
.
call_user_func($obj->callback);
우아하지는 않지만 …. @Gordon이 말하는 것은 아마도 유일한 길일 것입니다.
답변
글쎄, 당신이 정말로 주장 한다면 . 또 다른 해결 방법은 다음과 같습니다.
$obj = new ArrayObject(array(),2);
$obj->callback = function() {
print "HelloWorld!";
};
$obj['callback']();
그러나 그것은 가장 좋은 구문이 아닙니다.
그러나, PHP 파서는 항상 취급 T_OBJECT_OPERATOR
, IDENTIFIER
, (
메서드 호출한다. ->
메서드 테이블 을 우회하고 대신 속성에 액세스 할 수있는 해결 방법이없는 것 같습니다 .
답변
나는 이것이 오래되었다는 것을 알고 있지만 PHP 5.4 이상을 사용하는 경우 Traits 가이 문제를 잘 처리한다고 생각합니다.
먼저 속성을 호출 가능하게 만드는 특성을 만듭니다.
trait CallableProperty {
public function __call($method, $args) {
if (property_exists($this, $method) && is_callable($this->$method)) {
return call_user_func_array($this->$method, $args);
}
}
}
그런 다음 클래스에서 해당 특성을 사용할 수 있습니다.
class CallableStdClass extends stdClass {
use CallableProperty;
}
이제 익명 함수를 통해 속성을 정의하고 직접 호출 할 수 있습니다.
$foo = new CallableStdClass();
$foo->add = function ($a, $b) { return $a + $b; };
$foo->add(2, 2); // 4