匿名函数(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
可见,继承变量的行为是在函数定义时产生的。上例中定义func时,没有找到外部的func时,没有找到外部的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]," "; }; $msg = new ArrayObject([1,2,3], ArrayObject::ARRAY_AS_PROPS); $func(); ?> 运行输出 PHP Notice: Undefined variable: msg
我们改为使用引用传递
$func = function()use(&$msg){ echo $msg[0]," "; }; 运行输出 1
可见使用引用传递时,即使变量滞后于函数定义,函数内部还是可以找到外部相应的变量,不会出现变量未定义的情况。两者还是有区别的。
关于class中匿名函数里的this及use
<?php class C{ protected $_num = 0; public function mkFunc(){ $func = function(){ echo $this->_num++, " "; }; return $func; } public function get(){ echo $this->_num," "; } } $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++, " "; }; return $func; } 运行输出 PHP Fatal error: Cannot use $this as lexical variable
修改为
public function mkFunc(){ $self = $this; $func = function()use($self){ echo $this->_num++, " "; }; return $func; } 运行结果 0 1
可见是否使用use,效果是一样的。