如何通过PHP中的引用传递可变函数的参数?

时间:2010-04-10 03:11:10

标签: php pass-by-reference variadic-functions

假设有可能,如何在不在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来写;它的存在只是为了提供一个具体的例子。

2 个答案:

答案 0 :(得分:8)

从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;
}

答案 1 :(得分:6)

在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