PHP 7.4 中的预加载

本贴最后更新于 1318 天前,其中的信息可能已经斗转星移

在 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 将添加支持,这样您就不必自己制作预加载文件,并且只要您完全控制了此功能,就可以在服务器上轻松设置此功能。

  • PHP

    PHP(Hypertext Preprocessor)是一种开源脚本语言。语法吸收了 C 语言、 Java 和 Perl 的特点,主要适用于 Web 开发领域,据说是世界上最好的编程语言。

    179 引用 • 407 回帖 • 488 关注

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...
henryspace
迷失的人迷失了, 相逢的人会再相逢。 杭州

推荐标签 标签

  • API

    应用程序编程接口(Application Programming Interface)是一些预先定义的函数,目的是提供应用程序与开发人员基于某软件或硬件得以访问一组例程的能力,而又无需访问源码,或理解内部工作机制的细节。

    77 引用 • 430 回帖 • 1 关注
  • 996
    13 引用 • 200 回帖 • 6 关注
  • 锤子科技

    锤子科技(Smartisan)成立于 2012 年 5 月,是一家制造移动互联网终端设备的公司,公司的使命是用完美主义的工匠精神,打造用户体验一流的数码消费类产品(智能手机为主),改善人们的生活质量。

    4 引用 • 31 回帖 • 2 关注
  • NetBeans

    NetBeans 是一个始于 1997 年的 Xelfi 计划,本身是捷克布拉格查理大学的数学及物理学院的学生计划。此计划延伸而成立了一家公司进而发展这个商用版本的 NetBeans IDE,直到 1999 年 Sun 买下此公司。Sun 于次年(2000 年)六月将 NetBeans IDE 开源,直到现在 NetBeans 的社群依然持续增长。

    78 引用 • 102 回帖 • 683 关注
  • IBM

    IBM(国际商业机器公司)或万国商业机器公司,简称 IBM(International Business Machines Corporation),总公司在纽约州阿蒙克市。1911 年托马斯·沃森创立于美国,是全球最大的信息技术和业务解决方案公司,拥有全球雇员 30 多万人,业务遍及 160 多个国家和地区。

    17 引用 • 53 回帖 • 137 关注
  • Ant-Design

    Ant Design 是服务于企业级产品的设计体系,基于确定和自然的设计价值观上的模块化解决方案,让设计者和开发者专注于更好的用户体验。

    17 引用 • 23 回帖
  • Gzip

    gzip (GNU zip)是 GNU 自由软件的文件压缩程序。我们在 Linux 中经常会用到后缀为 .gz 的文件,它们就是 Gzip 格式的。现今已经成为互联网上使用非常普遍的一种数据压缩格式,或者说一种文件格式。

    9 引用 • 12 回帖 • 134 关注
  • 安全

    安全永远都不是一个小问题。

    199 引用 • 816 回帖
  • 周末

    星期六到星期天晚,实行五天工作制后,指每周的最后两天。再过几年可能就是三天了。

    14 引用 • 297 回帖
  • 资讯

    资讯是用户因为及时地获得它并利用它而能够在相对短的时间内给自己带来价值的信息,资讯有时效性和地域性。

    55 引用 • 85 回帖
  • Lute

    Lute 是一款结构化的 Markdown 引擎,支持 Go 和 JavaScript。

    25 引用 • 191 回帖 • 15 关注
  • 禅道

    禅道是一款国产的开源项目管理软件,她的核心管理思想基于敏捷方法 scrum,内置了产品管理和项目管理,同时又根据国内研发现状补充了测试管理、计划管理、发布管理、文档管理、事务管理等功能,在一个软件中就可以将软件研发中的需求、任务、bug、用例、计划、发布等要素有序的跟踪管理起来,完整地覆盖了项目管理的核心流程。

    6 引用 • 15 回帖 • 113 关注
  • Kotlin

    Kotlin 是一种在 Java 虚拟机上运行的静态类型编程语言,由 JetBrains 设计开发并开源。Kotlin 可以编译成 Java 字节码,也可以编译成 JavaScript,方便在没有 JVM 的设备上运行。在 Google I/O 2017 中,Google 宣布 Kotlin 成为 Android 官方开发语言。

    19 引用 • 33 回帖 • 65 关注
  • JetBrains

    JetBrains 是一家捷克的软件开发公司,该公司位于捷克的布拉格,并在俄国的圣彼得堡及美国麻州波士顿都设有办公室,该公司最为人所熟知的产品是 Java 编程语言开发撰写时所用的集成开发环境:IntelliJ IDEA

    18 引用 • 54 回帖 • 1 关注
  • 工具

    子曰:“工欲善其事,必先利其器。”

    286 引用 • 729 回帖
  • BAE

    百度应用引擎(Baidu App Engine)提供了 PHP、Java、Python 的执行环境,以及云存储、消息服务、云数据库等全面的云服务。它可以让开发者实现自动地部署和管理应用,并且提供动态扩容和负载均衡的运行环境,让开发者不用考虑高成本的运维工作,只需专注于业务逻辑,大大降低了开发者学习和迁移的成本。

    19 引用 • 75 回帖 • 642 关注
  • RYMCU

    RYMCU 致力于打造一个即严谨又活泼、专业又不失有趣,为数百万人服务的开源嵌入式知识学习交流平台。

    4 引用 • 6 回帖 • 52 关注
  • frp

    frp 是一个可用于内网穿透的高性能的反向代理应用,支持 TCP、UDP、 HTTP 和 HTTPS 协议。

    20 引用 • 7 回帖
  • Spark

    Spark 是 UC Berkeley AMP lab 所开源的类 Hadoop MapReduce 的通用并行框架。Spark 拥有 Hadoop MapReduce 所具有的优点;但不同于 MapReduce 的是 Job 中间输出结果可以保存在内存中,从而不再需要读写 HDFS,因此 Spark 能更好地适用于数据挖掘与机器学习等需要迭代的 MapReduce 的算法。

    74 引用 • 46 回帖 • 552 关注
  • Netty

    Netty 是一个基于 NIO 的客户端-服务器编程框架,使用 Netty 可以让你快速、简单地开发出一个可维护、高性能的网络应用,例如实现了某种协议的客户、服务端应用。

    49 引用 • 33 回帖 • 22 关注
  • Markdown

    Markdown 是一种轻量级标记语言,用户可使用纯文本编辑器来排版文档,最终通过 Markdown 引擎将文档转换为所需格式(比如 HTML、PDF 等)。

    167 引用 • 1513 回帖 • 1 关注
  • 数据库

    据说 99% 的性能瓶颈都在数据库。

    342 引用 • 708 回帖
  • Hibernate

    Hibernate 是一个开放源代码的对象关系映射框架,它对 JDBC 进行了非常轻量级的对象封装,使得 Java 程序员可以随心所欲的使用对象编程思维来操纵数据库。

    39 引用 • 103 回帖 • 709 关注
  • CSS

    CSS(Cascading Style Sheet)“层叠样式表”是用于控制网页样式并允许将样式信息与网页内容分离的一种标记性语言。

    198 引用 • 550 回帖 • 3 关注
  • Love2D

    Love2D 是一个开源的, 跨平台的 2D 游戏引擎。使用纯 Lua 脚本来进行游戏开发。目前支持的平台有 Windows, Mac OS X, Linux, Android 和 iOS。

    14 引用 • 53 回帖 • 530 关注
  • Pipe

    Pipe 是一款小而美的开源博客平台。Pipe 有着非常活跃的社区,可将文章作为帖子推送到社区,来自社区的回帖将作为博客评论进行联动(具体细节请浏览 B3log 构思 - 分布式社区网络)。

    这是一种全新的网络社区体验,让热爱记录和分享的你不再感到孤单!

    132 引用 • 1114 回帖 • 124 关注
  • 面试

    面试造航母,上班拧螺丝。多面试,少加班。

    325 引用 • 1395 回帖 • 1 关注