假设有可能,如何通过引用可变参数函数来传递参数而不在 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 需要所有参数都按引用传递。如果您按原样在 MySQLi_stmt_throwing
实例上调用 bind_result
,则参数将按值传递,并且不会进行绑定(bind)。
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 5.6 开始,您可以 pass arguments by reference到一个可变参数函数。这是 the RFC 中的示例:
public function prepare($query, &...$params) {
$stmt = $this->pdo->prepare($query);
foreach ($params as $i => &$param) {
$stmt->bindParam($i + 1, $param);
}
return $stmt;
}
关于php - 在 PHP 中如何通过引用传递可变参数函数的参数?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2612086/