Доступ к закрытым переменным из замыкания - PullRequest
12 голосов
/ 16 сентября 2010

Я пытаюсь сослаться на закрытую переменную объекта из замыкания. Код ниже, кажется, работает, но он жалуется Fatal error: Cannot access self:: when no class scope is active in test.php on line 12 и Fatal error: Using $this when not in object context in test.php on line 20.

Любые идеи, как добиться тех же результатов, используя замыкание, сохраняя переменные закрытыми и не создавая вспомогательных функций (опровергая саму идею закрытой переменной).

class MyClass
{

    static private $_var1;
    private $_var2;

    static function setVar1( $value )
    {
        $closure = function () use ( $value ) {
            self::$_var1 = $value;
        };
        $closure();
    }

    function setVar2( $value )
    {
        $closure = function () use ( $value ) {
            $this->_var2 = $value;
        };
        $closure();
    }

}

MyClass::setVar1( "hello" ); //doesn't work

$myclass = new MyClass;
$myclass->setVar2( "hello" ); //doesn't work

Ответы [ 3 ]

14 голосов
/ 16 сентября 2010

Изменить, чтобы заметить, Этот ответ изначально предназначался для PHP5.3 и ранее, теперь это возможно.Для получения текущей информации см. этот ответ .


Это невозможно напрямую.В частности, у замыканий нет связанной области, поэтому они не могут получить доступ к закрытым и защищенным элементам.

Однако вы можете использовать ссылки:

<?php
class MyClass
{

    static private $_var1;
    private $_var2;

    static function setVar1( $value )
    {
        $field =& self::$_var1;
        $closure = function () use ( $value,  &$field ) {
            $field = $value;
        };
        $closure();
    }

    function setVar2( $value )
    {
        $field =& $this->_var2;
        $closure = function () use ( $value, &$field ) {
            $field = $value;
        };
        $closure();
    }

}

MyClass::setVar1( "hello" );

$myclass = new MyClass;
$myclass->setVar2( "hello" );
4 голосов
/ 08 августа 2013

Это возможно, начиная с PHP 5.4.0

class test {
    function testMe() {
        $test = new test;
        $func = function() use ($test) {
            $test->findMe();        // Can see protected method
            $test::findMeStatically();  // Can see static protected method
        };
        $func();
        return $func;
    }

    protected function findMe() {
        echo " [find Me] \n";
    }

    protected static function findMeStatically() {
        echo " [find Me Statically] \n";
    }
}

$test = new test;
$func = $test->testMe();
$func();        // Can call from another context as long as 
            // the closure was created in the proper context.
2 голосов
/ 16 сентября 2010

У замыканий нет понятия $this или self - они не связаны с объектами таким образом.Это означает, что вам придется передавать переменные через предложение use ... что-то вроде:

$_var1 =& self::$_var1;
$closure = function() use ($value, &$_var1) {
  $_var1 = $value;
};

$_var2 =& $this->_var2;
$closure = function() use ($value, &$_var2) {
  $_var2 = $value;
};

Я не проверял приведенный выше код, но считаю, что он правильный.

...