Как можно передать аргументы в функции с переменным числом по ссылке в PHP? - PullRequest
13 голосов
/ 10 апреля 2010

Предполагая, что это возможно, как можно передать аргументы по ссылке на переменную функцию, не генерируя предупреждения в 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 ]

8 голосов
/ 27 октября 2014

Начиная с 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;
}
6 голосов
/ 10 апреля 2010

Нет способа передать списки аргументов переменной длины по ссылке в 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

...