解释的很详细,分享一下。
1 <?php 2 /** 3 * author: selfimpr 4 * mail: lgg860911@yahoo.com.cn 5 * blog: http://blog.csdn.net/lgg201 6 * 下面提到的代码在PHP5.3以上版本运行通过. 7 */ 8 function callback($callback) { 9 $callback(); 10 } 11 //输出: This is a anonymous function.<br />/n 12 //这里是直接定义一个匿名函数进行传递, 在以往的版本中, 这是不可用的. 13 //现在, 这种语法非常舒服, 和javascript语法基本一致, 之所以说基本呢, 需要继续向下看 14 //结论: 一个舒服的语法必然会受欢迎的. 15 callback(function() { 16 print "This is a anonymous function.<br />/n"; 17 }); 18 //输出: This is a closure use string value, msg is: Hello, everyone.<br />/n 19 //这里首先定义了一个闭包, 这次户口本上有名字了... 20 //use, 一个新鲜的家伙... 21 //众所周知, 闭包: 内部函数使用了外部函数中定义的变量. 22 //在PHP新开放的闭包语法中, 我们就是用use来使用闭包外部定义的变量的. 23 //这里我们使用了外部变量$msg, 定义完之后, 又对其值进行了改变, 闭包被执行后输出的是原始值 24 //结论: 以传值方式传递的基础类型参数, 闭包use的值在闭包创建是就确定了. 25 $msg = "Hello, everyone"; 26 $callback = function () use ($msg) { 27 print "This is a closure use string value, msg is: $msg. <br />/n"; 28 }; 29 $msg = "Hello, everybody"; 30 callback($callback); 31 //输出: This is a closure use string value lazy bind, msg is: Hello, everybody.<br />/n 32 //换一种引用方式, 我们使用引用的方式来use 33 //可以发现这次输出是闭包定义后的值... 34 //这个其实不难理解, 我们以引用方式use, 那闭包use的是$msg这个变量的地址 35 //当后面对$msg这个地址上的值进行了改变之后, 闭包内再输出这个地址的值时, 自然改变了. 36 $msg = "Hello, everyone"; 37 $callback = function () use (&$msg) { 38 print "This is a closure use string value lazy bind, msg is: $msg. <br />/n"; 39 }; 40 $msg = "Hello, everybody"; 41 callback($callback); 42 //输出: This is a closure use object, msg is: Hello, everyone.<br />/n 43 //闭包中输出的是之前被拷贝的值为Hello, everyone的对象, 后面是对$obj这个名字的一个重新赋值. 44 //可以这样考虑 45 //1. obj是对象Hello, everyone的名字 46 //2. 对象Hello, everyone被闭包use, 闭包产生了一个对Hello, everyone对象的引用 47 //3. obj被修改为Hello, everybody这个对象的名字 48 //4. 注意, 是名字obj代表的实体变了, 而不是Hello, everyone对象, 那自然闭包的输出还是前面的Hello, everyone 49 $obj = (object) "Hello, everyone"; 50 $callback = function () use ($obj) { 51 print "This is a closure use object, msg is: {$obj->scalar}. <br />/n"; 52 }; 53 $obj = (object) "Hello, everybody"; 54 callback($callback); 55 //输出: This is a closure use object, msg is: Hello, everybody.<br />/n 56 //还是按照上面的步骤, 按部就班的来吧: 57 //1. obj名字指向Hello, everyone对象 58 //2. 闭包产生一个引用指向Hello, everyone对象 59 //3. 修改obj名字指向的对象(即Hello, everyone对象)的scalar值 60 //4. 执行闭包, 输出的自然是Hello, everybody, 因为其实只有一个真正的对象 61 $obj = (object) "Hello, everyone"; 62 $callback = function () use ($obj) { 63 print "This is a closure use object, msg is: {$obj->scalar}. <br />/n"; 64 }; 65 $obj->scalar = "Hello, everybody"; 66 callback($callback); 67 //输出: This is a closure use object lazy bind, msg is: Hello, everybody.<br />/n 68 //闭包引用的是什么呢? &$obj, 闭包产生的引用指向$obj这个名字所指向的地址. 69 //因此, 无论obj怎么变化, 都是逃不脱的.... 70 //所以, 输出的就是改变后的值 71 $obj = (object) "Hello, everyone"; 72 $callback = function () use (&$obj) { 73 print "This is a closure use object lazy bind, msg is: {$obj->scalar}. <br />/n"; 74 }; 75 $obj = (object) "Hello, everybody"; 76 callback($callback); 77 /** 78 * 一个利用闭包的计数器产生器 79 * 这里其实借鉴的是python中介绍闭包时的例子... 80 * 我们可以这样考虑: 81 * 1. counter函数每次调用, 创建一个局部变量$counter, 初始化为1. 82 * 2. 然后创建一个闭包, 闭包产生了对局部变量$counter的引用. 83 * 3. 函数counter返回创建的闭包, 并销毁局部变量, 但此时有闭包对$counter的引用, 84 * 它并不会被回收, 因此, 我们可以这样理解, 被函数counter返回的闭包, 携带了一个游离态的 85 * 变量. 86 * 4. 由于每次调用counter都会创建独立的$counter和闭包, 因此返回的闭包相互之间是独立的. 87 * 5. 执行被返回的闭包, 对其携带的游离态变量自增并返回, 得到的就是一个计数器. 88 * 结论: 此函数可以用来生成相互独立的计数器. 89 */ 90 function counter() { 91 $counter = 1; 92 return function() use(&$counter) {return $counter ++;}; 93 } 94 $counter1 = counter(); 95 $counter2 = counter(); 96 echo "counter1: " . $counter1() . "<br />/n"; 97 echo "counter1: " . $counter1() . "<br />/n"; 98 echo "counter1: " . $counter1() . "<br />/n"; 99 echo "counter1: " . $counter1() . "<br />/n"; 100 echo "counter2: " . $counter2() . "<br />/n"; 101 echo "counter2: " . $counter2() . "<br />/n"; 102 echo "counter2: " . $counter2() . "<br />/n"; 103 echo "counter2: " . $counter2() . "<br />/n"; 104 ?>
文章来源:http://blog.csdn.net/lgg201/article/details/6127564