We can do black magic, which is useful in templating block calls:
<?php
$object->__named('methodNameHere', array('arg3' => 'three', 'arg1' => 'one'));
...
/**
* Pass method arguments by name
*
* @param string $method
* @param array $args
* @return mixed
*/
public function __named($method, array $args = array())
{
$reflection = new ReflectionMethod($this, $method);
$pass = array();
foreach($reflection->getParameters() as $param)
{
/* @var $param ReflectionParameter */
if(isset($args[$param->getName()]))
{
$pass[] = $args[$param->getName()];
}
else
{
$pass[] = $param->getDefaultValue();
}
}
return $reflection->invokeArgs($this, $pass);
}
?>
ReflectionMethod::invokeArgs
(PHP 5 >= 5.1.0)
ReflectionMethod::invokeArgs — Invoque les arguments
Description
Invoque la méthode réfléchie et lui passe les arguments sous la forme d'un tableau.
Liste de paramètres
-
object -
L'objet sur lequel invoquer la méthode. Si la méthode est statique, vous pouvez passer null pour ce paramètre.
-
args -
Les paramètres à passer à la méthode, sous forme de tableau.
Valeurs de retour
Retourne le résultat de la méthode.
Erreurs / Exceptions
Une ReflectionException si object
n'est pas une instance de la classe prévue pour cette méthode.
Une ReflectionException si l'invocation de la méthode échoue.
Exemples
Exemple #1 Exemple pour ReflectionMethod::invokeArgs()
<?php
class HelloWorld {
public function sayHelloTo($name) {
return 'Hello ' . $name;
}
}
$reflectionMethod = new ReflectionMethod('HelloWorld', 'sayHelloTo');
echo $reflectionMethod->invokeArgs(new HelloWorld(), array('Mike'));
?>
L'exemple ci-dessus va afficher :
Hello Mike
Notes
Note:
Si la fonction a des arguments qui ont besoin d'être des références, alors ils doivent être passés par références dans la liste des arguments.
Voir aussi
- ReflectionMethod::invoke() - Invoque
- __invoke()
- call_user_func_array() - Appelle une fonction de rappel avec les paramètres rassemblés en tableau
Passing arguments by reference works:
<?php $rm->invokeArgs($object, array(&$foo, $bar)); ?>
There is a simple workaround for the reference passing problem:
Since the reflection api has to handle all parameters in a generic way it has no chance to guess if you wish to pass data per value or reference.
But it seems that you can also decide to pass a reference from where you call the function or method (not just only by the ampersand prefix in its declaration).
So just do the following; which worked for me:
<?php
//...
$method->invoke($object, $inputValue, &$outputValue);
?>
Since this will only be necessary with arrays and primitive data types it should be acceptable in most cases to know in advance if you need to pass per reference. But it is probably although necessary to keep the ampersand always in the declaration (because of the at least two layers between the actual function and your invoke call).
If this is the expected behavior it will maybe make sense to mention it in the documentation for invoke and invokeArgs.
it seems that ReflectionMethod::invodeArgs() dont like reference-arguments in the method to call:
<?php
class Test{
function foo(&$arg){
$arg++;
}
}
$test = new Test();
$class = new ReflectionClass("Test");
$method = $class->getMethod("foo");
$bar = 9;
$method->invokeArgs($test, array($bar));
?>
RESULT: "Invocation of method Test::foo() failed"
