zoukankan      html  css  js  c++  java
  • php 5.3新增的闭包语法介绍function() use() {}

    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. ?>  
  • 相关阅读:
    write to logfile
    open and read a file content to a variable
    strategy
    Android 开机启动程序
    消息队列
    卡机音乐功能实现
    Android 2.0 开机动画文件分析
    多线程实例
    消息队列
    多线程实例
  • 原文地址:https://www.cnblogs.com/raobenjun/p/7424656.html
Copyright © 2011-2022 走看看