Предполагая, что это возможно, как передать аргументы ссылкой на переменную функцию без генерации предупреждения в 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
; он существует просто для того, чтобы привести конкретный пример.
Невозможно передавать переменные аргументов переменной длины по ссылке в 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
Начиная с 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; }