IT技术博客大学习 共学习 共进步

深入理解PHP之匿名函数

风雪之隅 2010-06-24 09:45:27 浏览 3,565 次

PHP中, 传递Callback的方式, 一直很丑陋. 在PHP5.3以前, 我们只有俩种选择:

1. 字符串的函数名2. 使用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_func7. 返回新的函数

我们来验证下:

<?phpcreate_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出来查看.

而在PHP5.3发布的时候, 其中有一条new feature就是支持闭包/Lambda Function, 我第一反应是以为zval新增了一个IS_FUNCTION, 但实际上是构造了一个PHP5.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)?>

而PHP5.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对闭包的支持比起来, 还是有些太简陋了~

建议继续学习

  1. 不定参数的应用 function(fmt, …) (阅读 4,744)
  2. 函数式编程 (阅读 4,585)
  3. JavaScript的5种调用函数的方法 (阅读 4,446)
  4. C 语言中统一的函数指针 (阅读 4,187)
  5. C语言函数实现的另类方法 (阅读 3,904)
  6. 关于在函数调用时传递string引用的必要性 (阅读 3,884)
  7. javascript匿名函数 (阅读 3,446)
  8. 情人节特献:有心之函数必然就有分手函数 (阅读 3,363)
  9. MySQL 内部函数简介 (阅读 3,322)
  10. JavaScript 函数、作用域和继承 (阅读 3,285)