闭包函数是创建时,封装周围状态的函数,而匿名函数是没有名称的函数,匿名函数可以被赋值给变量,也就是所谓的函数式编程,也可以传递参数,经常作为回调函数.(理论上讲:匿名函数和闭包不算是一个概念,php却视为一个概念).
列举一个闭包
<?php $greet = function ($name) { return sprintf("Hello %s ", $name); }; echo $greet('LaravelAcademy.org');
结果:
Hello LaravelAcademy.org
闭包和普通的函数很像,都能接受参数和返回值,只是闭包没有函数名(匿名函数),$greet这个变量是一个闭包函数,而且闭包对象实现了__invoke魔术方法,只要变量名后有(),PHP就会查找调用__invoke.php有很多内置函数都是用的闭包,例如 array_map(),这是使用PHP匿名函数的绝佳时机。要记住,闭包和其他值一样,可以作为参数传入其他PHP函数
<?php $numberList= array_map(function ($number) {
return $number += 1;
}, [1, 2, 3]);
print_r($numberList);
之前的随笔http://www.cnblogs.com/hanshuai0921/p/6761908.html中提到JS和python使用闭包可以访问到闭包函数外面的变量,也就是一个函数可以访问到外部的变量
但是在php中使用父作用域的变量和状态必须手动调用闭包对象的bindTo
方法或使用use
关键字
在laravel框架底层使用事务将模型数据保存到数据库
我们使用闭包保存返回的状态,使用use获取父作用域的$option,这样闭包函数才能访问到$option的数据
传递多个父作用域变量到闭包
闭包应该可以被称为对象,所以我们可以在闭包中使用$this关键字获取闭包的内部状态,闭包对象的默认状态没什么用,需要注意的是其中的__invoke
魔术方法和bindTo
方法。
__invoke
的作用是当尝试以调用函数的方式调用一个对象时,__invoke()
方法会被自动调用。
接下来我们来看看bindTo
方法,通过该方法,我们可以把闭包的内部状态绑定到其他对象上。这里bindTo
方法的第二个参数显得尤为重要,其作用是指定绑定闭包的那个对象所属的PHP类,这样,闭包就可以在其他地方访问邦定闭包的对象中受保护和私有的成员变量。
你会发现,PHP框架经常使用bindTo
方法把路由URL映射到匿名回调函数上,框架会把匿名回调函数绑定到应用对象上,这样在匿名函数中就可以使用$this
关键字引用重要的应用对象:
这些变量在匿名函数被销毁前,一直保存着自己的状态,这两个结果都是1,是因为PHP默认是通过复制的方式传入父作用域变量,如果想改变副作用域的变量值,就要引用的方式传递use (&$i).所以上面的结果是1和1