Прежде всего, я хочу проверить функцию:
private function testMe (array &$output)
{
$output['a'] = 3; // $$$$ $output gets changes
}
Я сделал небольшой метод, чтобы сделать его публичным, и позвонил:
public static function makePublicAndCall ($objectInstance, $methodname)
{
$ref = new ReflectionMethod (get_class($objectInstance), $methodname);
$ref->setAccessible(true);
$params = array();
for ($i = 2+1; $i <= func_num_args(); $i++)
{
$params[] = func_get_arg($i-1);
}
$result = $ref->invokeArgs ($objectInstance, $params);
for ($i = 2+1; $i <= func_num_args(); $i++)
{
// write back $$$$ here I would need something like "func_get_arg($i-1)"}
return $result;
}
Итак, используя его:
$output = array();
::makePublicAndCall ($object, 'testMe', $output);
// $output OMG output remains the same! It must not be empty but [a] => 3
видите проблему? Этот метод имеет 2 обязательных параметра, а все остальные являются необязательными (они идут к самому вызываемому методу). Но если эти параметры изменены, их нельзя будет перенести обратно!
В PHP 5.6 введены переменные аргументы, которые также могут принимать параметры по ссылке.
function makePublicAndCall ($objectInstance, $methodname, &...$args) { }
теперь просто вперед $args
массив заполнен аргументами по ссылке на $objectInstance->$methodname
function makePublicAndCall ($objectInstance, $methodname, &...$args) {
$ref = new ReflectionMethod (get_class($objectInstance), $methodname);
$ref->setAccessible(true);
return $ref->invokeArgs($objectInstance, $args);
}
makePublicAndCall($object, 'testMe', $output);
Нет, кстати, прости.
Передача времени обращения по ссылке все еще работает с этими архаичными версиями, поэтому не стесняйтесь использовать ее.
function makePublicAndCall ($objectInstance, $methodname) {
$ref = new ReflectionMethod (get_class($objectInstance), $methodname);
$ref->setAccessible(true);
return $ref->invokeArgs ($objectInstance, $args);
}
@makePublicAndCall($object, 'testMe', &$output); // note the & here...
Кроме того, вам не нужно ожидать ссылки в вашем testMe
функция, вы получаете массив, заполненный ссылками, этого достаточно; вам не нужно получать массив, заполненный ссылками, для работы с ссылками.
Я решил это независимо от версий:
public static function makePublicAndCall ($objectInstance, $methodname, array &$params = array())
{
$ref = new ReflectionMethod (get_class($objectInstance), $methodname);
$ref->setAccessible(true);
return $ref->invokeArgs ($objectInstance, array(&$params));
}
призвание:
::makePublicAndCall ($object, 'testMe', $output);
РЕДАКТИРОВАТЬ: другое решение независимо от версии, но держу пари, что вы бросите свою шляпу на пол:
public static function makeStaticMethodPublicAndCall ($className, $method, &$params1 = null, &$params2 = null, &$params3 = null, &$params4 = null)
{
$className = new ReflectionClass($className);
$method = $className->getMethod($method);
$method->setAccessible(true);
return $method->invokeArgs (null, array(&$params1, &$params2, &$params3, &$params4));
}
.
.
.
::makeStaticMethodPublicAndCall ('MyClass', 'myMethod', $param1, $param2);