PHP 闭包我的的简单理解,就是在匿名函数中经过use直接使用函数以外的变量。闭包
这个函数以外如何定义?函数
例如scala
<? function total(){ $total = 0; $sub = 0; $callback = function($a, $b) use ($total, &$sub){ //$total 即匿名函数以外的变量 //&$sub 即匿名函数以外的引用 $sub = $a+$b+$total; }; $callback(1, 2); return $sub; } ?>
PHP 5.3以后引入了新的闭包语法 即function() use (){}code
function能够添加参数做为匿名函数的调用参数对象
use能够添加参数和引用,这些参数位于匿名函数以外,经过use来使用。string
具体语法以下:io
//定义一个回掉函数 function callback($callback) { $callback(); } //这里是直接定义一个匿名函数进行传递, 在以往的版本中, 这是不可用的. callback(function() { print "This is a anonymous function."; }); //输出: This is a anonymous function. //这里首先定义了一个闭包, 此次户口本上有名字了... //use, 一个新鲜的家伙... //众所周知, 闭包: 内部函数使用了外部函数中定义的变量. //在PHP新开放的闭包语法中, 咱们就是用use来使用闭包外部定义的变量的. //这里咱们使用了外部变量$msg, 定义完以后, 又对其值进行了改变, 闭包被执行后输出的是原始值 //结论: 以传值方式传递的基础类型参数, 闭包use的值在闭包建立是就肯定了. $msg = "Hello, everyone"; $callback = function () use ($msg) { print "This is a closure use string value, msg is: $msg. "; }; $msg = "Hello, everybody"; callback($callback); //输出: This is a closure use string value, msg is: Hello, everyone. //换一种引用方式, 咱们使用引用的方式来use //能够发现此次输出是闭包定义后的值... //这个其实不难理解, 咱们以引用方式use, 那闭包use的是$msg这个变量的地址 //当后面对$msg这个地址上的值进行了改变以后, 闭包内再输出这个地址的值时, 天然改变了. $msg = "Hello, everyone"; $callback = function () use (&$msg) { print "This is a closure use string value lazy bind, msg is: $msg. "; }; $msg = "Hello, everybody"; callback($callback); //输出: This is a closure use string value lazy bind, msg is: Hello, everybody. //闭包中输出的是以前被拷贝的值为Hello, everyone的对象, 后面是对$obj这个名字的一个从新赋值. //能够这样考虑 //1. obj是对象Hello, everyone的名字 //2. 对象Hello, everyone被闭包use, 闭包产生了一个对Hello, everyone对象的引用 //3. obj被修改成Hello, everybody这个对象的名字 //4. 注意, 是名字obj表明的实体变了, 而不是Hello, everyone对象, 那天然闭包的输出仍是前面的Hello, everyone $obj = (object) "Hello, everyone"; $callback = function () use ($obj) { print "This is a closure use object, msg is: {$obj->scalar}. "; }; $obj = (object) "Hello, everybody"; callback($callback); //输出: This is a closure use object, msg is: Hello, everyone. //仍是按照上面的步骤, 循序渐进的来吧: //1. obj名字指向Hello, everyone对象 //2. 闭包产生一个引用指向Hello, everyone对象 //3. 修改obj名字指向的对象(即Hello, everyone对象)的scalar值 //4. 执行闭包, 输出的天然是Hello, everybody, 由于其实只有一个真正的对象 $obj = (object) "Hello, everyone"; $callback = function () use ($obj) { print "This is a closure use object, msg is: {$obj->scalar}. "; }; $obj->scalar = "Hello, everybody"; callback($callback); //输出: This is a closure use object, msg is: Hello, everybody. //闭包引用的是什么呢? &$obj, 闭包产生的引用指向$obj这个名字所指向的地址. //所以, 不管obj怎么变化, 都是逃不脱的.... //因此, 输出的就是改变后的值 $obj = (object) "Hello, everyone"; $callback = function () use (&$obj) { print "This is a closure use object lazy bind, msg is: {$obj->scalar}. "; }; $obj = (object) "Hello, everybody"; callback($callback); //输出: This is a closure use object lazy bind, msg is: Hello, everybody.