假设可能的话,如何在不产生 PHP 警告的情况下将参数按引用传递给可变函数?我们不能再在函数调用中使用 '&' 运算符,否则我会接受它(即使这样做容易出错,如果编码人员忘记了它)。
这启发了我挖掘旧的 MySQLi 包装器类(现在,我只会使用 PDO)。包装器和 MySQLi 类之间唯一的区别是包装器抛出异常而不是返回 FALSE。
这启发了我挖掘旧的 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
,则参数将以值传递并且绑定操作将无法执行。
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
为例;它存在只是为了提供一个具体的例子。