从PHP 7.4.0开始,可以将PHP配置为在引擎启动时将脚本预加载到opcache中。然后,这些文件中的任何符号(函数,类等)将对所有请求全局可用,而无需明确包含。这会为基线内存使用量交换便利性和性能(因为代码始终可用)。它还需要重新启动PHP进程以清除预加载的脚本,这意味着此功能仅在生产环境中有效,而在开发环境中则不可用。
请注意,性能和内存之间的最佳平衡可能会因应用程序而异。“预装所有内容”可能是最简单的策略,但不一定是最佳策略。此外,仅当从一个请求到另一个请求存在持久过程时,预加载才有用。这意味着,如果启用了操作缓存,虽然它可以在CLI脚本中运行,但通常毫无意义。在FFI库上使用预加载时例外。
注意事项:
Windows不支持预加载。
配置预加载涉及两个步骤,并且需要启用opcache。首先, 在php.ini中设置opcache.preload值:
opcache.preload = preload.php
preload.php是一个任意文件,它将在服务器启动时运行一次(PHP-FPM,mod_php等),并将代码加载到永久内存中。如果PHP将以root身份运行(不建议),则opcache.preload_user 值可以指定备用系统用户来运行预加载。不允许以root身份运行预加载。
在preload.php脚本中,include, include_once,require,require_once或 opcache_compile_file()引用的任何文件都将解析为持久内存。在以下示例中,除非是文件,否则src目录中的所有.php文件都将被预加载。 Test
<?php $directory = new RecursiveDirectoryIterator(__DIR__ . '/src'); $fullTree = new RecursiveIteratorIterator($directory); $phpFiles = new RegexIterator($fullTree, '/.+((?<!Test)+\.php$)/i', RecursiveRegexIterator::GET_MATCH); foreach ($phpFiles as $key => $file) { require_once($file[0]); } ?>
双方包括和opcache_compile_file()将工作,但有代码被如何处理不同的含义。
include将执行文件中的代码,而 opcache_compile_file()则不会。这意味着只有前者支持条件声明(在if块内声明的函数)。
因为include将执行代码,所以嵌套的include d文件也将被解析并预先加载其声明。
opcache_compile_file()可以按任何顺序加载文件。也就是说,如果 a.php只会定义类
A
和 b.php定义类B
延伸A
,然后 opcache_compile_file()可以以任何顺序加载这些两个文件。但是,使用 include时,必须首先包含 a.php 。在任何一种情况下,如果更高版本的脚本包含已经预加载的文件,则其内容仍将执行,但是将不会重新定义其定义的任何符号。使用 include_once不会阻止文件被第二次包含。
因此,哪种方法更好取决于所需的行为。使用否则会使用自动加载器的代码,opcache_compile_file()可以提供更大的灵活性。使用本来可以手动加载的代码,include将更加健壮。
https://www.php.net/manual/zh/opcache.preloading.php