1. 程式人生 > 其它 >php 匿名函式中的use

php 匿名函式中的use

匿名函式(Anonymous functions),也叫閉包函式(closures),

允許 臨時建立一個沒有指定名稱的函式。最經常用作回撥函式callable引數的值。當然,也有其它應用的情況。

匿名函式目前是通過Closure類來實現的。

閉包可以從父作用域中繼承變數,任何此類變數都應該用use語言結構傳遞進去。

PHP 7.1 起,不能傳入此類變數:superglobals$this或者和引數重名。

匿名函式中的use,其作用就是從父作用域繼承變數。
下例是最常見的用法,如果不使用use,函式中將找不到變數$msg。

<?php
$msg = [1,2,3];
$func = function()use($msg){
    print_r($msg);
};  

$func();
?> 執行輸出 Array ( [0] => 1 [1] => 2 [2] => 3 )

關於繼承變數的時機

繼承變數的行為是在函式定義時產生還是在函式呼叫時產生?我們調整下上例中程式碼的順序,將$msg置於函式定義之後。

<?php
$func = function()use($msg){
    print_r($msg);
};  

$msg = [1,2,3];

$func();
?>

執行輸出
PHP Notice:  Undefined variable: msg in test.php on line 4

可見,繼承變數的行為是在函式定義時產生的。上例中定義

funcfunc時,沒有找到外部的msg,所以函式執行時$msg就是未定義變數。

關於use中使用引用傳值

我們知道,在匿名函式的use中如果使用引用傳值,那麼匿名函式中對引數值的改變會同樣影響外部相應變數。比如下面的例子:

<?php
$msg = [1,2,3];
$func = function()use(&$msg){
    $msg[0]++;
    print_r($msg);
};


$func();

print_r($msg);
?>

執行輸出
Array
(
    [0] => 2
    [1] => 2
    [
2] => 3 ) Array ( [0] => 2 [1] => 2 [2] => 3 )

那麼是不是任何情況下,想通過匿名函式改變外部變數值都一定要通過引用方式向use傳值呢?看下面這個例子:

<?php
$msg = new ArrayObject([1,2,3], ArrayObject::ARRAY_AS_PROPS);
$func = function()use($msg){
    $msg[0]++;
    print_r($msg);
};

$func();
print_r($msg);
?>

執行輸出
ArrayObject Object
(
    [storage:ArrayObject:private] => Array
        (
            [0] => 2
            [1] => 2
            [2] => 3
        )

)
ArrayObject Object
(
    [storage:ArrayObject:private] => Array
        (
            [0] => 2
            [1] => 2
            [2] => 3
        )

)

可見,如果傳遞object型別的變數,即使不顯示使用引用傳遞,匿名函式中變數值的改變同樣會影響到外部相關變數。

但是,問題又來了。向use傳遞object變數時,使用引用與不使用引用到底有沒有區別呢?還是來看例子

<?php
$func = function()use($msg){
    echo $msg[0],"\n";
};

$msg = new ArrayObject([1,2,3], ArrayObject::ARRAY_AS_PROPS);
$func();
?>

執行輸出
PHP Notice:  Undefined variable: msg

我們改為使用引用傳遞

$func = function()use(&$msg){
    echo $msg[0],"\n";
};

執行輸出
1

可見使用引用傳遞時,即使變數滯後於函式定義,函式內部還是可以找到外部相應的變數,不會出現變數未定義的情況。兩者還是有區別的。

關於class中匿名函式裡的this及use

<?php
class C{
    protected $_num = 0;

    public function mkFunc(){
        $func = function(){
            echo $this->_num++, "\n";
        };

        return $func;
    }

    public function get(){
        echo $this->_num,"\n";
    }
}

$obj = new C();
$func = $obj->mkFunc();
$func();

$obj->get();
?>

執行結果
0
1

可見匿名函式裡的this就是指當前物件,不需要使用use就可以直接找到。

還是上面的例子,如果一定要使用use會是什麼效果呢?
將mkFunc改為

public function mkFunc(){
    //唯一改動是此處加了use
    $func = function()use($this){
        echo $this->_num++, "\n";
    };

    return $func;
}

執行輸出
PHP Fatal error:  Cannot use $this as lexical variable 

修改為

public function mkFunc(){
    $self = $this;
    $func = function()use($self){
        echo $this->_num++, "\n";
    };

    return $func;
}

執行結果
0
1

可見是否使用use,效果是一樣的。