了解抽象语法树

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

如何做…

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

  1. function test()
  2. {
  3. return [
  4. 1 => function () { return [
  5. 1 => function ($a) { return 'Level 1/1:' . ++$a; },
  6. 2 => function ($a) { return 'Level 1/2:' . ++$a; },
  7. ];},
  8. 2 => function () { return [
  9. 1 => function ($a) { return 'Level 2/1:' . ++$a; },
  10. 2 => function ($a) { return 'Level 2/2:' . ++$a; },
  11. ];}
  12. ];
  13. }
  14. $a = 't';
  15. $t = 'test';
  16. 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。 在构造函数中,我们构建并定义两个数组。 第一个数组由过滤器回调组成。 第二个数组具有验证回调:

  1. public function __construct()
  2. {
  3. $this->filter = [
  4. 'striptags' => function ($a) { return strip_tags($a); },
  5. 'digits' => function ($a) { return preg_replace(
  6. '/[^0-9]/', '', $a); },
  7. 'alpha' => function ($a) { return preg_replace(
  8. '/[^A-Z]/i', '', $a); }
  9. ];
  10. $this->validate = [
  11. 'alnum' => function ($a) { return ctype_alnum($a); },
  12. 'digits' => function ($a) { return ctype_digit($a); },
  13. 'alpha' => function ($a) { return ctype_alpha($a); }
  14. ];
  15. }

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

  1. $security->filterDigits($item));

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

  1. public function __call($method, $params)
  2. {
  3. preg_match('/^(filter|validate)(.*?)$/i', $method, $matches);
  4. $prefix = $matches[1] ?? '';
  5. $function = strtolower($matches[2] ?? '');
  6. if ($prefix && $function) {
  7. return $this->$prefix[$function]($params[0]);
  8. }
  9. return $value;
  10. }

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

{% hint style=”info” %} 不要对这些东西太着迷了!

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

如何运行…

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

  1. require __DIR__ . '/../Application/Autoload/Loader.php';
  2. Application\Autoload\Loader::init(__DIR__ . '/..');
  3. $security = new Application\Web\Security();

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

  1. $data = [
  2. '<ul><li>Lots</li><li>of</li><li>Tags</li></ul>',
  3. 12345,
  4. 'This is a string',
  5. 'String with number 12345',
  6. ];

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

  1. foreach ($data as $item) {
  2. echo 'ORIGINAL: ' . $item . PHP_EOL;
  3. echo 'FILTERING' . PHP_EOL;
  4. printf('%12s : %s' . PHP_EOL,'Strip Tags', $security->filterStripTags($item));
  5. printf('%12s : %s' . PHP_EOL, 'Digits', $security->filterDigits($item));
  6. printf('%12s : %s' . PHP_EOL, 'Alpha', $security->filterAlpha($item));
  7. echo 'VALIDATORS' . PHP_EOL;
  8. printf('%12s : %s' . PHP_EOL, 'Alnum',
  9. ($security->validateAlnum($item)) ? 'T' : 'F');
  10. printf('%12s : %s' . PHP_EOL, 'Digits',
  11. ($security->validateDigits($item)) ? 'T' : 'F');
  12. printf('%12s : %s' . PHP_EOL, 'Alpha',
  13. ($security->validateAlpha($item)) ? 'T' : 'F');
  14. }

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

了解抽象语法树 - 图1

参考

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