匿名函数,允许临时创建一个没有名称的函数。大多数情况下是作为回调函数的参数使用。当然,也有其它应用的场景。
匿名函数的使用
在 PHP 中,匿名函数实际上就是一个 Closure对象:
var_dump(function () {}); // object(Closure)[2]
Closure对象中定义了魔术方法 __invoke() ,当对象被当做函数调用时,会自动地调用这个 __invoke()方法。
可以把它传递给一个变量,这就相当于把一个对象赋给了变量,因此可以通过该变量调用匿名函数:
$foo = function () {echo 1;};
$foo(); // 1
在大部分情况下,匿名函数是作为匿名函数的参数使用的:
$callback = function ($val, $key) {
echo $key . ':' . $val . '<br/>';
};
$arr = ['name' => 'yyy', 'sex' => 'male'];
array_walk($arr, $callback);
也可以直接:
array_walk($arr, function ($val, $key) {
echo $key . ':' . $val . '<br/>';
});
/*
上例输出:
name:yyy
sex:male
*/
从父作用域中继承变量
因为 PHP 中,函数中定义的变量默认为局部变量,因此在 PHP中,使用父作用域(定义匿名函数的地方)的变量之前,都必须用 use 语言结构将父作用域中的变量传递进去。可以按值传递也可以按引用传递,默认为按值传递。
值得注意的是, 从父作用域继承的变量,值在匿名函数创建的时候就已经了,而不是在调用时确定,如:
function parent() {
$val = 1;
$child = function () use ($val) {
echo $val;
};
$val = 2;
$child();
}
parent();
/*
上例输出:
1
*/
另外,从父作用域中继承变量与使用全局变量是不同的。全局变量存在于一个全局的范围,而匿名函数的父作用域是定义该匿名函数的地方。并且当同时使用了 global声明和 use时,匿名函数中的变量会优先被设置为全局变量。
$val = 1;
function parent() {
$val = 2;
$child = function () use ($val) {
global $val;
echo $val;
};
$child();
}
parent();
/*
上例输出:
1
*/
闭包
先看代码:
function getCounter() {
$i = 0;
return function() use($i) { // 这里应该使用引用传入变量: use(&$i)
echo ++$i;
};
}
$counter = getCounter();
$counter(); // 1
$counter(); // 1
和 Javascript中不同,这里两次函数调用并没有使$i变量自增,默认PHP是通过拷贝的方式传入上层变量进入匿名函数, 如果需要改变上层变量的值则需要通过引用的方式传递。
$counter变量指向了 getCounter()内部的匿名函数,因此 getCounter()函数里的变量的生存期变成了静态生存期,这些变量在匿名函数被销毁之前始终在内存中维持着原来的值。
但是注意,闭包只是使得变量的生存期变成静态了,而作用域还是局部的,因此只能通过匿名函数来访问这些局部变量。