PHP匿名函数

匿名函数,允许临时创建一个没有名称的函数。大多数情况下是作为回调函数的参数使用。当然,也有其它应用的场景。

匿名函数的使用

在 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()函数里的变量的生存期变成了静态生存期,这些变量在匿名函数被销毁之前始终在内存中维持着原来的值。

但是注意,闭包只是使得变量的生存期变成静态了,而作用域还是局部的,因此只能通过匿名函数来访问这些局部变量。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值