匿名函數中的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中如果使用引用傳值,那麼匿名函數中對參數值的改變會同樣影響外部對應變數。例如下面的例子:
<?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
可見使用引用傳遞時,即使變數落後於函數定義,函數內部還是可以找到外部對應的變量,不會出現變數未定義的情況。兩者還是有差別的。
<?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中文網其他相關文章!