作为一名开发人员,不受 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
。 在构造函数中,我们构建并定义两个数组。 第一个数组由过滤器回调组成。 第二个数组具有验证回调:
public function __construct()
{
$this->filter = [
'striptags' => function ($a) { return strip_tags($a); },
'digits' => function ($a) { return preg_replace(
'/[^0-9]/', '', $a); },
'alpha' => function ($a) { return preg_replace(
'/[^A-Z]/i', '', $a); }
];
$this->validate = [
'alnum' => function ($a) { return ctype_alnum($a); },
'digits' => function ($a) { return ctype_digit($a); },
'alpha' => function ($a) { return ctype_alpha($a); }
];
}
4.我们希望能够以一种对开发者友好的方式来调用这个功能。因此,如果我们想过滤数字,那么最好是运行这样的命令:
$security->filterDigits($item));
为此,我们定义了魔术方法 __call()
,它使我们可以访问不存在的方法:
public function __call($method, $params)
{
preg_match('/^(filter|validate)(.*?)$/i', $method, $matches);
$prefix = $matches[1] ?? '';
$function = strtolower($matches[2] ?? '');
if ($prefix && $function) {
return $this->$prefix[$function]($params[0]);
}
return $value;
}
我们使用 preg_match()
将 $method
参数与 filter
或 validate
进行匹配。 然后,第二个子匹配项将在 $this->filter
或 $this-> validate
中转换为数组键。 如果两个子模式都产生子匹配项,则将第一个子匹配项分配给 $prefix
,将第二个子匹配项 $function
分配给 $prefix
。 当执行适当的回调时,这些最终将作为变量参数。
{% hint style="info" %} 不要对这些东西太着迷了!
当你沉浸在AST所带来的新的表达自由中时,一定要记住,你最终写的代码,从长远来看,可能是非常“神秘“的,这最终会造成长期的维护问题。 {% endhint %}
首先,我们创建一个示例文件 chap_02_web_filtering_ast_example.php
,利用第一章 "建立基础 "中定义的自动加载类,获取 Application\Web\Security
的实例:
require __DIR__ . '/../Application/Autoload/Loader.php';
Application\Autoload\Loader::init(__DIR__ . '/..');
$security = new Application\Web\Security();
接下来,我们定义一个测试数据块:
$data = [
'<ul><li>Lots</li><li>of</li><li>Tags</li></ul>',
12345,
'This is a string',
'String with number 12345',
];
最后,我们为每个测试数据项调用每个过滤器和验证器:
foreach ($data as $item) {
echo 'ORIGINAL: ' . $item . PHP_EOL;
echo 'FILTERING' . PHP_EOL;
printf('%12s : %s' . PHP_EOL,'Strip Tags', $security->filterStripTags($item));
printf('%12s : %s' . PHP_EOL, 'Digits', $security->filterDigits($item));
printf('%12s : %s' . PHP_EOL, 'Alpha', $security->filterAlpha($item));
echo 'VALIDATORS' . PHP_EOL;
printf('%12s : %s' . PHP_EOL, 'Alnum',
($security->validateAlnum($item)) ? 'T' : 'F');
printf('%12s : %s' . PHP_EOL, 'Digits',
($security->validateDigits($item)) ? 'T' : 'F');
printf('%12s : %s' . PHP_EOL, 'Alpha',
($security->validateAlpha($item)) ? 'T' : 'F');
}
下面是一些输入字符串的输出:
关于AST的更多信息,请查阅针对抽象语法树的RFC,可以在https://wiki.php.net/rfc/abstract_syntax_tree 上看到。