您好,欢迎访问一九零五行业门户网

深入理解PHP中的匿名函数

匿名函数的作用就是扩大函数的使用功能,在php 5.3以前,传递callback的方式,我们只有俩种选择:
◆字符串的函数名
◆使用create_function的返回
在php5.3以后, 我们多了一个选择, 也就是closure。





$func = function () { ... };  
array_walk($arr, $func);  从实现上来说, 第一种方式: 传递函数名字符串是最简单的。而第二种方式create_function, 其实和第一种方式本质上一样的, create_function返回一个字符串的函数名, 这个函数名的格式是:





\000_lambda_ . count(anonymous_functions)++ 
我们来看看create_function的实现步骤:
1. 获取参数, 函数体;
2. 拼凑一个function __lambda_func (参数) { 函数体;} 的字符串;
3. eval;
4. 通过__lambda_func在函数表中找到eval后得到的函数体, 找不到就出错;
5. 定义一个函数名:\000_lambda_ . count(anonymous_functions)++;
6. 用新的函数名替换__lambda_func;
7. 返回新的函数。
我们来验证下:





php 
create_function(, 'echo __function__;');   call_user_func(\000lambda_1, 1);   ?>  //输出   __lambda_fun  因为在eval的时候, 函数名是”__lambda_func”, 所以匿名函数内会输出__lambda_func, 而因为最后用”\000_lambda_”.count(anonymous_functions)++重命名了函数表中的”__lambda_func”函数, 所以可通过”\000_lambda_” . count(anonymous_functions)++调用这个匿名函数。为了证实这一点, 可以将create_function的返回值dump出来查看。
而在php 5.3发布的时候, 其中有一条new feature就是支持闭包/lambda function, 我第一反应是以为zval新增了一个is_function, 但实际上是构造了一个php 5.3引入的closure”类”的实例, closure类的构造函数是私有的, 所以不能被直接实例化, 另外closure类是final类, 所以也不能做为基类派生子类.





//php-5.3.0  

$class = new reflectionclass(closure);   var_dump($class->isinternal());   var_dump($class->isabstract() );   var_dump($class->isfinal());   var_dump($class->isinterface());   //输出:   bool(true)   bool(false)   bool(true)   bool(false)   ?>  而php 5.3中对闭包的支持, 也仅仅是把要保持的外部变量, 做为closure对象的”static属性”(并不是普通意义上的可遍历/访问的属性).





//php-5.3.0  

$b = laruence;   $func = function($a) use($b) {};   var_dump($func);   /* 输出:   object(closure)#1 (2) {   [static]=>  array(1) {   [b]=>  string(8) laruence   }   [parameter]=>  array(1) {   [$a]=>  string(10) required>   }   }   */  这个实现, 个人认为和js对闭包的支持比起来, 还是有些太简陋了。
其它类似信息

推荐信息