Press "Enter" to skip to content

深入理解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属性"(并不是普通意义上的可遍历/访问的属性).

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

15 Comments

  1. Alvin Tang
    Alvin Tang February 10, 2022

    鸟哥,如果是开发PHP扩展,怎么使用闭包呢?

  2. daniel
    daniel May 15, 2018

    如果这个闭包函数写在类中,是用var_dump打印这个闭包的话;
    在和static属性同级别的数组中还有个this属性,这是代表当前的对象吗?

  3. Exceptional post however , I was wondering if you could write a litte more on this subject?
    I’d be very thankful if you could elaborate a little bit
    further. Kudos!

  4. Farmer18Concepcion
    Farmer18Concepcion September 4, 2010

    One understands that men’s life seems to be not cheap, but different people require money for different stuff and not every one earns big sums cash. Therefore to receive some loan or consolidation loans should be a proper way out.

  5. […] 转自: http://www.laruence.com/2010/06/20/1602.html 发表评论 | Trackback 2010年8月3日 | 归档于 编程高手 标签: PHP, 匿名函数, 闭包 « MyISAM和InnoDB两种引擎的区别 本文目前尚无任何评论. […]

  6. wynn
    wynn June 22, 2010

    Closure机制的编写者在RFC里有提到这个问题,基本意思就是说这方面不是PHP的核心机制,因此不会在这方面下太多力气。

  7. 神仙
    神仙 June 21, 2010

    底子差,没办法。

  8. ElmerZhang
    ElmerZhang June 21, 2010

    不好意思,是我看错了,呵呵

  9. 雪候鸟
    雪候鸟 June 20, 2010

    @ElmerZhang 没有啊? 你是指哪里错了?

  10. ElmerZhang
    ElmerZhang June 20, 2010

    第二种方式create_function 的代码贴错了吧?

Comments are closed.