【问题标题】:How can arguments to variadic functions be passed by reference in PHP?如何在 PHP 中通过引用传递可变参数函数的参数?
【发布时间】:2011-02-06 09:54:06
【问题描述】:

假设有可能,如何通过引用可变参数函数来传递参数而不在 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;它的存在只是为了提供一个具体的例子。

【问题讨论】:

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


【解决方案1】:

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

【讨论】:

    【解决方案2】:

    在 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(&amp;$x) 导致变量通过引用传递,而不是函数签名。

    来自对 PHP 函数参数文档的评论:http://php.net/manual/en/functions.arguments.php

    【讨论】:

    • 这在 PHP 5.6 中是否随着可变参数的引入而改变?
    • @Mark:有这样的问题,最好查阅PHP手册。然后,有了这些信息,您就可以发布新的答案了。
    • @outis 文档实际上并没有说。 The RFC推荐了,不知道有没有正式实现。
    • @Mark:您可能需要更仔细地阅读“函数参数”页面。在“Variable-length argument lists ... in PHP 5.6+”部分的末尾,它指出:“最后,您还可以通过在 ... 前加上与号 (&) 来通过引用传递变量参数。”
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2019-02-12
    • 2021-12-11
    • 1970-01-01
    • 2018-06-26
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多