background image

PHP 匿名函数的理解

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_func
7. 返回新的函数名我们来验证下:
<?php
create_function("", 'echo __FUNCTION__;');
call_user_func("\000lambda_1", 1);
?>
//输出
__lambda_func 因 为在 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 属

”(并不是普通意义上的可遍历/访问的属性).