首頁 後端開發 PHP7 PHP 7.4中的預先載入(Opcache Preloading)

PHP 7.4中的預先載入(Opcache Preloading)

Nov 30, 2019 pm 02:04 PM
PHP 7.4

在PHP 7.4中,添加了对预加载的支持,这是一个可以显著提高代码性能的特性。

简而言之,这是它的工作方式:

● 为了预加载文件,您需要编写一个自定义PHP脚本

● 该脚本在服务器启动时执行一次

● 所有预加载的文件在内存中都可用于所有请求

● 在重新启动服务器之前,对预加载文件所做的更改不会产生任何影响

让我们深入了解它。

#Opcache

虽然预加载是建立在opcache之上的,但它并不是完全一样的。Opcache将获取您的PHP源文件,将其编译为“ opcodes”,然后将这些编译后的文件存储在磁盘上。

您可以将操作码看作是代码的底层表示,在运行时很容易解释。因此,opcache会跳过源文件和PHP解释器在运行时实际需要之间的转换步骤。巨大的胜利!

但我们还有更多的收获。Opcached文件不知道其他文件。如果类a是从类B扩展而来的,那么仍然需要在运行时将它们链接在一起。此外,opcache执行检查以查看源文件是否被修改,并将基于此使其缓存失效。

因此,这就是预加载发挥作用的地方:它不仅将源文件编译为操作码,而且还将相关的类、特征和接口链接在一起。然后,它将这个“已编译”的可运行代码blob(即:PHP解释器可以使用的代码)保存在内存中。

现在,当请求到达服务器时,它可以使用已经加载到内存中的部分代码库,而不会产生任何开销。

那么,我们所说的“代码库的一部分”是什么呢?

#实践中的预加载

为了进行预加载,开发人员必须告知服务器要加载哪些文件。这是用一个简单的PHP脚本完成的,确实没有什么困难。

规则很简单:

● 您提供一个预加载脚本,并使用opcache.preload命令将其链接到您的php.ini文件中。

● 您要预加载的每个PHP文件都应该传递到opcache_compile_file(),或者在预加载脚本中只需要一次。

假设您想要预加载一个框架,例如Laravel。您的脚本必须遍历vendor/laravel目录中的所有PHP文件,并将它们一个接一个地添加。

在php.ini中链接到此脚本的方法如下:

opcache.preload=/path/to/project/preload.php
登入後複製

这是一个虚拟的实现:

$files = /* An array of files you want to preload */;
foreach ($files as $file) {
    opcache_compile_file($file);
}
登入後複製

#警告:无法预加载未链接的类

等等,有一个警告!为了预加载文件,还必须预加载它们的依赖项(接口,特征和父类)。

如果类依赖项有任何问题,则会在服务器启动时通知您:

Can't preload unlinked class 
Illuminate\Database\Query\JoinClause: 
Unknown parent 
Illuminate\Database\Query\Builder
登入後複製

看,opcache_compile_file()将解析一个文件,但不执行它。这意味着如果一个类有未预加载的依赖项,它本身也不能预加载。

这不是一个致命的问题,您的服务器可以正常工作。但你不会得到所有你想要的预加载文件。

幸运的是,还有一种确保链接文件也被加载的方法:您可以使用require_once代替opcache_compile_file,让已注册的autoloader(可能是composer的)负责其余的工作。

$files = /* All files in eg. vendor/laravel */;
foreach ($files as $file) {
    require_once($file);
}
登入後複製

还有一些需要注意的地方。例如,如果您试图预加载Laravel,那么框架中的一些类依赖于其他尚不存在的类。例如,文件系统缓存类\ lighting \ filesystem \ cache依赖于\League\Flysystem\Cached\Storage\AbstractCache,如果您从未使用过文件系统缓存,则可能无法将其安装到您的项目中。

尝试预加载所有内容时,您可能会遇到“class not found”错误。幸运的是,在默认的Laravel安装中,只有少数这些类,可以轻易忽略。为了方便起见,我编写了一个小小的preloader类,以使忽略文件更容易,如下所示:

class Preloader
{
    private array $ignores = [];
    private static int $count = 0;
    private array $paths;
    private array $fileMap;
    public function __construct(string ...$paths)
    {
        $this->paths = $paths;
        // We'll use composer's classmap
        // to easily find which classes to autoload,
        // based on their filename
        $classMap = require __DIR__ . '/vendor/composer/autoload_classmap.php';
        $this->fileMap = array_flip($classMap);
    }
    
    public function paths(string ...$paths): Preloader
    {
        $this->paths = array_merge(
            $this->paths,
            $paths
        );
        return $this;
    }
    public function ignore(string ...$names): Preloader
    {
        $this->ignores = array_merge(
            $this->ignores,
            $names
        );
        return $this;
    }
    public function load(): void
    {
        // We'll loop over all registered paths
        // and load them one by one
        foreach ($this->paths as $path) {
            $this->loadPath(rtrim($path, '/'));
        }
        $count = self::$count;
        echo "[Preloader] Preloaded {$count} classes" . PHP_EOL;
    }
    private function loadPath(string $path): void
    {
        // If the current path is a directory,
        // we'll load all files in it 
        if (is_dir($path)) {
            $this->loadDir($path);
            return;
        }
        // Otherwise we'll just load this one file
        $this->loadFile($path);
    }
    private function loadDir(string $path): void
    {
        $handle = opendir($path);
        // We'll loop over all files and directories
        // in the current path,
        // and load them one by one
        while ($file = readdir($handle)) {
            if (in_array($file, ['.', '..'])) {
                continue;
            }
            $this->loadPath("{$path}/{$file}");
        }
        closedir($handle);
    }
    private function loadFile(string $path): void
    {
        // We resolve the classname from composer's autoload mapping
        $class = $this->fileMap[$path] ?? null;
        // And use it to make sure the class shouldn't be ignored
        if ($this->shouldIgnore($class)) {
            return;
        }
        // Finally we require the path,
        // causing all its dependencies to be loaded as well
        require_once($path);
        self::$count++;
        echo "[Preloader] Preloaded `{$class}`" . PHP_EOL;
    }
    private function shouldIgnore(?string $name): bool
    {
        if ($name === null) {
            return true;
        }
        foreach ($this->ignores as $ignore) {
            if (strpos($name, $ignore) === 0) {
                return true;
            }
        }
        return false;
    }
}
登入後複製

通过在相同的预加载脚本中添加此类,我们现在可以像这样加载整个Laravel框架:

// …
(new Preloader())
    ->paths(__DIR__ . '/vendor/laravel')
    ->ignore(
        \Illuminate\Filesystem\Cache::class,
        \Illuminate\Log\LogManager::class,
        \Illuminate\Http\Testing\File::class,
        \Illuminate\Http\UploadedFile::class,
        \Illuminate\Support\Carbon::class,
    )
    ->load();
登入後複製

#有效吗?

这当然是最重要的问题:所有文件都正确加载了吗?您可以简单地通过重新启动服务器来测试它,然后将opcache_get_status()的输出转储到PHP脚本中。您将看到它有一个名为preload_statistics的键,它将列出所有预加载的函数、类和脚本;以及预加载文件消耗的内存。

# Composer支持

一个很有前途的特性可能是基于composer的自动预加载解决方案,它已经被大多数现代PHP项目所使用。人们正在努力在composer.json中添加预加载配置选项,它将为您生成预加载文件!目前,此功能仍在开发中,但您可以在此处关注。

#服务器要求

在使用预加载时,关于devops方面还有两件更重要的事情需要提及。

您已經知道,需要在php.ini中指定一個條目才能預先載入。這意味著如果您使用共享主機,您將無法自由地配置PHP。實際上,您需要一個專用的(虛擬)伺服器,以便能夠為單一專案最佳化預先載入的檔案。記住這一點。

也要記住,每次需要重新載入記憶體檔案時,都需要重新啟動伺服器(如果使用php-fpm就足夠了)。這對大多數人來說似乎是顯而易見的,但仍然值得一提。

#效能

現在到最重要的問題:預先載入真的能提高效能嗎?

答案是肯定的:Ben Morel分享了一些基準測試,可以在之前連結的相同的composer問題中找到。

有趣的是,您可以決定僅預先載入“hot classes”,它們是程式碼庫中經常使用的類別。 Ben的基準測試顯示,只載入約100個熱門類,實際上可以獲得比預先載入所有類別更好的效能效益。這是性能提升13%和17%的差別。

當然,應該預先載入哪些類別取決於您的特定項目。明智的做法是在開始時盡可能預先加載。如果您確實需要少量的百分比增長,您將不得不在運行時監視您的程式碼。

當然,所有這些工作都可以自動化,將來可能會實現。

現在,最重要的是要記住composer將添加支持,這樣您就不必自己製作預加載文件,並且只要您完全控制了此功能,就可以在伺服器上輕鬆設置此功能。 

譯:https://stitcher.io/blog/preloading-in-php-74

#

以上是PHP 7.4中的預先載入(Opcache Preloading)的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

熱門話題

Java教學
1653
14
CakePHP 教程
1413
52
Laravel 教程
1306
25
PHP教程
1251
29
C# 教程
1224
24