一个趋势是,开发者开始使用匿名函数。当处理匿名函数时,一个典型的问题是,如何将它们写成任何对象都可以绑定到 $this
的形式,并且函数仍然可以工作。在 PHP 5 代码中使用的方法是使用 bindTo()
。在 PHP 7 中,增加了一个新的方法 call()
,它提供了类似的功能,但性能有很大的提升。
如何做...
为了利用 call()
,在一个冗长的循环中执行一个匿名函数。在这个例子中,我们将演示一个匿名函数,它可以扫描一个日志文件,根据出现的频率识别 IP 地址:
1.首先,我们定义一个 Application\Web\Access
类。 在构造函数中,我们接受文件名作为参数。日志文件通过 SplFileObject
打开,并分配给 $this->log
:
Namespace Application\Web;
use Exception;
use SplFileObject;
class Access
{
const ERROR_UNABLE = 'ERROR: unable to open file';
protected $log;
public $frequency = array();
public function __construct($filename)
{
if (!file_exists($filename)) {
$message = __METHOD__ . ' : ' . self::ERROR_UNABLE . PHP_EOL;
$message .= strip_tags($filename) . PHP_EOL;
throw new Exception($message);
}
$this->log = new SplFileObject($filename, 'r');
}
2.接下来,我们定义一个生成器,逐行遍历文件:
public function fileIteratorByLine()
{
$count = 0;
while (!$this->log->eof()) {
yield $this->log->fgets();
$count++;
}
return $count;
}
3. 最后,我们定义了一个寻找并提取一个IP地址作为子匹配的方法:
public function getIp($line)
{
preg_match('/(\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3})/', $line, $match);
return $match[1] ?? '';
}
}
如何运行...
首先,我们定义一个调用程序 chap_02_performance_using_php7_enchancement_call.php
,该程序利用第一章《建立基础》中定义的自动加载类来获取 Application\Web\Access
的实例:
define('LOG_FILES', '/var/log/apache2/*access*.log');
require __DIR__ . '/../Application/Autoload/Loader.php';
Application\Autoload\Loader::init(__DIR__ . '/..');
接下来我们定义匿名函数,处理日志文件中的一行。如果检测到一个IP地址,它就会成为 $frequency
数组中的一个键,并且这个键的当前值会递增:
// 定义方法
$freq = function ($line) {
$ip = $this->getIp($line);
if ($ip) {
echo '.';
$this->frequency[$ip] =
(isset($this->frequency[$ip])) ? $this->frequency[$ip] + 1 : 1;
}
};
然后,我们在每个找到的日志文件中循环迭代行,处理IP地址:
foreach (glob(LOG_FILES) as $filename) {
echo PHP_EOL . $filename . PHP_EOL;
// 存取类
$access = new Application\Web\Access($filename);
foreach ($access->fileIteratorByLine() as $line) {
$freq->call($access, $line);
}
}
小贴士
实际上在 PHP 5 中也可以做同样的事情,但是需要两行代码。
$func = $freq->bindTo($access);
$func($line);
性能比在 PHP 7 中使用 call()
慢20%到50%。
最后,我们对数组进行反向排序,但保留键。然后通过简单的 foreach()
循环输出:
arsort($access->frequency);
foreach ($access->frequency as $key => $value) {
printf('%16s : %6d' . PHP_EOL, $key, $value);
}
他的输出会根据你处理的 access.log
而有所不同。下面是一个例子:
更多...
许多 PHP 7 的性能改进与新特性和功能无关。相反,它们采取的是内部改进的形式,在开始运行程序之前是看不见的。以下是属于这一类的改进的简短列表:
| | |
| | 在 PHP 5 中,每次调用函数时都要对提供给函数的参数进行解析。参数以字符串的形式传入,并以类似于 scanf() 函数的方式进行解析。在 PHP 7 中,这个过程得到了优化,变得更加高效,从而使性能得到了显著的提高。这种改进很难衡量,但似乎在6%左右。 |
| | PHP NG(下一代)计划代表了PHP语言的大部分重写。它保留了现有的功能,但包含了所有可以想象到的节省时间和提高效率的措施。数据结构被压缩,内存被更有效地使用。仅仅是一个影响数组处理的变化,就使性能得到了显著的提高,同时大大减少了内存的使用。 |
| | 大约有二十多个扩展属于这些类别中的一种:废弃的、不再维护的、未维护的依赖关系、或未移植到 PHP 7。经过核心开发人员的投票,决定删除 2/3 或 "短名单 "上的扩展。这样做的结果是减少了开销,加快了PHP语言未来的整体开发速度。 |