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

问题描述 投票:13回答:2

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

php pass-by-reference variadic-functions
2个回答
7
投票

在 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.netmanualenfunctions.arguments.php


9
投票

在PHP 5.6中,你可以 可以 传话 到一个变量函数。下面是一个来自 指南针:

public function prepare($query, &...$params) {
    $stmt = $this->pdo->prepare($query);
    foreach ($params as $i => &$param) {
        $stmt->bindParam($i + 1, $param);
    }
    return $stmt;
}
© www.soinside.com 2019 - 2024. All rights reserved.