Как аргументы переменных переменных передаются по ссылке в PHP?
Предполагая, что возможно, как передать аргументы ссылкой на переменную функцию без генерации предупреждения в PHP? Мы больше не можем использовать '&' оператор в вызове функции, в противном случае я бы согласился с этим (хотя это было бы ошибкой, если кодер забыл об этом).
Что вдохновило это старые классы оболочки MySQLi, которые я раскопал (в наши дни я просто использовал PDO). Единственное различие между обертками и классами MySQLi - это исключение обертки, а не возврат FALSE
.
class DBException extends RuntimeException {}
...
class MySQLi_throwing extends mysqli {
...
function prepare($query) {
$stmt = parent::prepare($query);
if (!$stmt) {
throw new DBException($this->error, $this->errno);
}
return new MySQLi_stmt_throwing($this, $query, $stmt);
}
}
// I don't remember why I switched from extension to composition, but
// it shouldn't matter for this question.
class MySQLi_stmt_throwing /* extends MySQLi_stmt */ {
protected $_link, $_query, $_delegate;
public function __construct($link, $query, $prepared) {
//parent::__construct($link, $query);
$this->_link = $link;
$this->_query = $query;
$this->_delegate = $prepared;
}
function bind_param($name, &$var) {
return $this->_delegate->bind_param($name, $var);
}
function __call($name, $args) {
//$rslt = call_user_func_array(array($this, 'parent::' . $name), $args);
$rslt = call_user_func_array(array($this->_delegate, $name), $args);
if (False === $rslt) {
throw new DBException($this->_link->error, $this->errno);
}
return $rslt;
}
}
Трудность заключается в вызовах методов, таких как bind_result
на обертке. Функции константы (например, bind_param
) могут быть явно определены, что позволяет передавать по ссылке. bind_result
, однако, все аргументы должны быть переданы по ссылке. Если вы вызываете bind_result
в экземпляре MySQLi_stmt_throwing
as-is, аргументы передаются по значению и привязка не будет выполняться.
try {
$id = Null;
$stmt = $db->prepare('SELECT id FROM tbl WHERE ...');
$stmt->execute()
$stmt->bind_result($id);
// $id is still null at this point
...
} catch (DBException $exc) {
...
}
Так как вышеупомянутые классы больше не используются, этот вопрос является всего лишь вопросом любопытства. Альтернативные подходы к классам обертки не актуальны. Определение метода с кучей аргументов с использованием значений Null
по умолчанию неверно (что, если вы определяете 20 аргументов, но функция вызывается с 21?). Ответы даже не нужно писать в терминах MySQL_stmt_throwing
; он существует просто для предоставления конкретного примера.
Ответы
Ответ 1
Невозможно передавать переменные аргументов переменной длины по ссылке в PHP. Это фундаментальное ограничение языка.
Существует, однако, обходной путь с синтаксисом array(&$var1, &$var2...)
:
<?php
/** raise all our arguments to the power of 2 */
function pow2() {
$args = &func_get_arg(0);
for ($i = 0; $i< count($args); ++$i) {
$args[$i] *= 2;
}
}
$x = 1; $y = 2; $z = 3;
pow2(array(&$x, &$y, &$z)); // this is the important line
echo "$x, $y, $z"; // output "2, 4, 6"
?>
Test
также может быть объявлен function test($args)
, но я хотел бы проиллюстрировать, что это работает с семейством функций func_get_args()
. Это array(&$x)
, которое заставляет переменную передаваться по ссылке, а не подписи функции.
Из комментария к документации PHP по аргументам функции: http://php.net/manual/en/functions.arguments.php
Ответ 2
Начиная с PHP 5.6 вы можете передать аргументы по ссылке в вариационную функцию. Вот пример из RFC:
public function prepare($query, &...$params) {
$stmt = $this->pdo->prepare($query);
foreach ($params as $i => &$param) {
$stmt->bindParam($i + 1, $param);
}
return $stmt;
}