首頁 >後端開發 >php教程 >php匿名函數中的use有什麼用

php匿名函數中的use有什麼用

怪我咯
怪我咯原創
2017-06-28 11:43:282161瀏覽

匿名函數中的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 /search/ballqiu/c.php on line 4

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

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();
?>

我們改為使用引用傳遞

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

运行输出1

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


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

<?phpclass 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();?>
运行结果01

可見匿名函數裡的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;
}

运行结果01

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

以上是php匿名函數中的use有什麼用的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn