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