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

13
假设可能的话,如何在不产生 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,则参数将以值传递并且绑定操作将无法执行。

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为例;它存在只是为了提供一个具体的例子。


糟糕...刚刚发现这个问题是重复的:https://dev59.com/B3I-5IYBdhLWcg3wTWZu,尽管我更喜欢下面meager的答案而不是其他问题的被接受的答案。 - outis
2个回答

11

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

5

在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 "1, 4, 9"
Test也可以声明为function test($args),但我想说明这适用于func_get_args()函数系列。是array(&$x)导致变量被引用传递,而不是函数签名。
从一条有关PHP函数参数文档的评论中得知:http://php.net/manual/en/functions.arguments.php

随着可变参数的引入,PHP 5.6中是否有所改变? - mpen
@Mark:对于这样的问题,最好查看PHP手册。然后,掌握了这些信息,你可以发表新答案。 - outis
@outis文档实际上没有说明。RFC建议这样做,但我不知道它是否被纳入官方实现。 - mpen
@Mark:你可能需要更仔细地阅读“函数参数”页面。在“PHP 5.6+中的可变长度参数列表”部分的末尾,它指出:“最后,你也可以通过在省略号前加上一个&符号来传递可变参数的引用。” - outis

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接