了解抽象语法树

作为一名开发人员,不受 PHP 5和更早版本中强加的某些语法限制可能会让您感兴趣。除了前面提到的语法的一致性之外,语法方面的最大改进是调用任何返回值的能力,只需添加一组括号即可调用。此外,当返回值为数组时,您还可以直接访问任何数组元素。

如何做...

1.任何返回回调的函数或方法都可以通过简单地添加括号 () 来立即执行(有或没有参数)。通过简单地使用方括号 [];来表示元素,可以立即从任何返回数组的函数或方法中导出元素。在下面这个简短(但琐碎)的例子中,函数 test() 返回一个数组。数组包含六个匿名函数。$a 的值为 $t$$a 被解释为 $test

function test()
{
    return [
        1 => function () { return [
            1 => function ($a) { return 'Level 1/1:' . ++$a; },
            2 => function ($a) { return 'Level 1/2:' . ++$a; },
        ];},
        2 => function () { return [
            1 => function ($a) { return 'Level 2/1:' . ++$a; },
            2 => function ($a) { return 'Level 2/2:' . ++$a; },
        ];}
    ];
}

$a = 't';
$t = 'test';
echo $$a()[1]()[2](100);

2.AST允许我们发出 echo $$a()[1]()[2](100) 这样的命令。这是由左到右的解析,其执行方式如下:

  • $$a() 解释为 test(),它返回一个数组

  • [1] 派生数组元素 1,返回回调

  • () 执行这个回调,它返回一个由两个元素组成的数组

  • [2] 派生数组元素 2 ,返回回调

  • (100) 执行这个回调,提供一个 100 的值,返回 Level 1/2:101

小贴士

这样的语句在 PHP 5 中是不可能的:会返回一个解析错误。

3.下面是一个更实质性的示例,该示例利用AST语法定义数据过滤和验证类。 首先,我们定义Application\Web\Securityclass。 在构造函数中,我们构建并定义两个数组。 第一个数组由过滤器回调组成。 第二个数组具有验证回调:

4.我们希望能够以一种对开发者友好的方式来调用这个功能。因此,如果我们想过滤数字,那么最好是运行这样的命令:

为此,我们定义了魔术方法 __call(),它使我们可以访问不存在的方法:

我们使用 preg_match()$method 参数与 filtervalidate 进行匹配。 然后,第二个子匹配项将在 $this->filter$this-> validate 中转换为数组键。 如果两个子模式都产生子匹配项,则将第一个子匹配项分配给 $prefix,将第二个子匹配项 $function 分配给 $prefix。 当执行适当的回调时,这些最终将作为变量参数。

不要对这些东西太着迷了!

当你沉浸在AST所带来的新的表达自由中时,一定要记住,你最终写的代码,从长远来看,可能是非常“神秘“的,这最终会造成长期的维护问题。

如何运行...

首先,我们创建一个示例文件 chap_02_web_filtering_ast_example.php,利用第一章 "建立基础 "中定义的自动加载类,获取 Application\Web\Security 的实例:

接下来,我们定义一个测试数据块:

最后,我们为每个测试数据项调用每个过滤器和验证器:

下面是一些输入字符串的输出:

参考

关于AST的更多信息,请查阅针对抽象语法树的RFC,可以在https://wiki.php.net/rfc/abstract_syntax_tree 上看到。

最后更新于